blob: 591c4533eb31415c37f6a51e8ceaf5df99b74d03 [file] [log] [blame]
// Copyright 2017 The Cobalt Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include "starboard/shared/speechd/speechd_internal.h"
#if defined(ADDRESS_SANITIZER)
// By default, Leak Sanitizer and Address Sanitizer is expected to exist
// together. However, this is not true for all platforms.
// HAS_LEAK_SANTIZIER=0 explicitly removes the Leak Sanitizer from code.
#ifndef HAS_LEAK_SANITIZER
#define HAS_LEAK_SANITIZER 1
#endif // HAS_LEAK_SANITIZER
#endif // defined(ADDRESS_SANITIZER)
#if HAS_LEAK_SANITIZER
#include <sanitizer/lsan_interface.h>
#endif // HAS_LEAK_SANITIZER
#include "starboard/common/once.h"
#include "starboard/shared/starboard/application.h"
namespace starboard {
namespace shared {
namespace speechd {
namespace {
SbOnceControl init_once = SB_ONCE_INITIALIZER;
} // namespace
SpeechDispatcher* SpeechDispatcher::instance = NULL;
// static
SpeechDispatcher* SpeechDispatcher::Get() {
bool result = SbOnce(&init_once, &SpeechDispatcher::Initialize);
SB_DCHECK(result);
SB_DCHECK(instance);
return instance;
}
SpeechDispatcher::SpeechDispatcher() {
const char* client_name = "starboard_application";
#if HAS_LEAK_SANITIZER
// spd_open leaks memory even though spd_close is eventually called.
// Chromium's text-to-speech system for linux ran into the same issue:
// http://crbug.com/317360
__lsan_disable();
#endif // HAS_LEAK_SANITIZER
connection_ = spd_open(client_name, NULL, NULL, SPD_MODE_THREADED);
#if HAS_LEAK_SANITIZER
__lsan_enable();
#endif // HAS_LEAK_SANITIZER
if (!connection_) {
SB_DLOG(ERROR) << "Failed to initialize SpeechDispatcher.";
}
if (!SetLanguage(SbSystemGetLocaleId())) {
SB_DLOG(ERROR) << "Unable to set language to current locale.";
}
}
SpeechDispatcher::~SpeechDispatcher() {
SB_DLOG(INFO) << "Closing Speech Dispatcher.";
if (connection_) {
spd_close(connection_);
}
}
bool SpeechDispatcher::SetLanguage(const char* language) {
ScopedLock lock(lock_);
if (connection_) {
int result = spd_set_language(connection_, language);
return result == 0;
}
return false;
}
void SpeechDispatcher::Speak(const char* text) {
ScopedLock lock(lock_);
if (connection_ && text && *text) {
// Priority SPD_MESSAGE will be queued with other text of same priority.
int result = spd_say(connection_, SPD_MESSAGE, text);
if (result < 0) {
SB_DLOG(ERROR) << "Failed to speak: " << text;
}
}
}
void SpeechDispatcher::Cancel() {
ScopedLock lock(lock_);
if (connection_) {
int result = spd_cancel(connection_);
if (result != 0) {
SB_DLOG(ERROR) << "Failed to cancel speech.";
}
}
}
// static
void SpeechDispatcher::Initialize() {
starboard::Application* application = starboard::Application::Get();
if (application) {
instance = new SpeechDispatcher();
application->RegisterTeardownCallback(SpeechDispatcher::Destroy);
}
}
// static
void SpeechDispatcher::Destroy() {
delete instance;
instance = NULL;
}
} // namespace speechd
} // namespace shared
} // namespace starboard