• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef CONTENT_BROWSER_SPEECH_SPEECH_RECOGNITION_MANAGER_IMPL_H_
6 #define CONTENT_BROWSER_SPEECH_SPEECH_RECOGNITION_MANAGER_IMPL_H_
7 
8 #include <map>
9 #include <string>
10 
11 #include "base/basictypes.h"
12 #include "base/callback.h"
13 #include "base/compiler_specific.h"
14 #include "base/memory/weak_ptr.h"
15 #include "content/browser/renderer_host/media/media_stream_requester.h"
16 #include "content/public/browser/speech_recognition_event_listener.h"
17 #include "content/public/browser/speech_recognition_manager.h"
18 #include "content/public/browser/speech_recognition_session_config.h"
19 #include "content/public/browser/speech_recognition_session_context.h"
20 #include "content/public/common/speech_recognition_error.h"
21 
22 namespace media {
23 class AudioManager;
24 }
25 
26 namespace content {
27 class BrowserMainLoop;
28 class MediaStreamManager;
29 class MediaStreamUIProxy;
30 class SpeechRecognitionManagerDelegate;
31 class SpeechRecognizer;
32 
33 // This is the manager for speech recognition. It is a single instance in
34 // the browser process and can serve several requests. Each recognition request
35 // corresponds to a session, initiated via |CreateSession|.
36 //
37 // In any moment, the manager has a single session known as the primary session,
38 // |primary_session_id_|.
39 // This is the session that is capturing audio, waiting for user permission,
40 // etc. There may also be other, non-primary, sessions living in parallel that
41 // are waiting for results but not recording audio.
42 //
43 // The SpeechRecognitionManager has the following responsibilities:
44 //  - Handles requests received from various render views and makes sure only
45 //    one of them accesses the audio device at any given time.
46 //  - Handles the instantiation of SpeechRecognitionEngine objects when
47 //    requested by SpeechRecognitionSessions.
48 //  - Relays recognition results/status/error events of each session to the
49 //    corresponding listener (demuxing on the base of their session_id).
50 //  - Relays also recognition results/status/error events of every session to
51 //    the catch-all snoop listener (optionally) provided by the delegate.
52 class CONTENT_EXPORT SpeechRecognitionManagerImpl :
NON_EXPORTED_BASE(SpeechRecognitionManager)53     public NON_EXPORTED_BASE(SpeechRecognitionManager),
54     public SpeechRecognitionEventListener {
55  public:
56   // Returns the current SpeechRecognitionManagerImpl or NULL if the call is
57   // issued when it is not created yet or destroyed (by BrowserMainLoop).
58   static SpeechRecognitionManagerImpl* GetInstance();
59 
60   // SpeechRecognitionManager implementation.
61   virtual int CreateSession(
62       const SpeechRecognitionSessionConfig& config) OVERRIDE;
63   virtual void StartSession(int session_id) OVERRIDE;
64   virtual void AbortSession(int session_id) OVERRIDE;
65   virtual void AbortAllSessionsForRenderProcess(int render_process_id) OVERRIDE;
66   virtual void AbortAllSessionsForRenderView(int render_process_id,
67                                              int render_view_id) OVERRIDE;
68   virtual void StopAudioCaptureForSession(int session_id) OVERRIDE;
69   virtual const SpeechRecognitionSessionConfig& GetSessionConfig(
70       int session_id) const OVERRIDE;
71   virtual SpeechRecognitionSessionContext GetSessionContext(
72       int session_id) const OVERRIDE;
73   virtual int GetSession(int render_process_id,
74                          int render_view_id,
75                          int request_id) const OVERRIDE;
76   virtual bool HasAudioInputDevices() OVERRIDE;
77   virtual base::string16 GetAudioInputDeviceModel() OVERRIDE;
78   virtual void ShowAudioInputSettings() OVERRIDE;
79 
80   // SpeechRecognitionEventListener methods.
81   virtual void OnRecognitionStart(int session_id) OVERRIDE;
82   virtual void OnAudioStart(int session_id) OVERRIDE;
83   virtual void OnEnvironmentEstimationComplete(int session_id) OVERRIDE;
84   virtual void OnSoundStart(int session_id) OVERRIDE;
85   virtual void OnSoundEnd(int session_id) OVERRIDE;
86   virtual void OnAudioEnd(int session_id) OVERRIDE;
87   virtual void OnRecognitionEnd(int session_id) OVERRIDE;
88   virtual void OnRecognitionResults(
89       int session_id, const SpeechRecognitionResults& result) OVERRIDE;
90   virtual void OnRecognitionError(
91       int session_id, const SpeechRecognitionError& error) OVERRIDE;
92   virtual void OnAudioLevelsChange(int session_id, float volume,
93                                    float noise_volume) OVERRIDE;
94 
95   SpeechRecognitionManagerDelegate* delegate() const { return delegate_.get(); }
96 
97  protected:
98   // BrowserMainLoop is the only one allowed to istantiate and free us.
99   friend class BrowserMainLoop;
100   // Needed for dtor.
101   friend struct base::DefaultDeleter<SpeechRecognitionManagerImpl>;
102   SpeechRecognitionManagerImpl(media::AudioManager* audio_manager,
103                                MediaStreamManager* media_stream_manager);
104   virtual ~SpeechRecognitionManagerImpl();
105 
106  private:
107   // Data types for the internal Finite State Machine (FSM).
108   enum FSMState {
109     SESSION_STATE_IDLE = 0,
110     SESSION_STATE_CAPTURING_AUDIO,
111     SESSION_STATE_WAITING_FOR_RESULT,
112     SESSION_STATE_MAX_VALUE = SESSION_STATE_WAITING_FOR_RESULT
113   };
114 
115   enum FSMEvent {
116     EVENT_ABORT = 0,
117     EVENT_START,
118     EVENT_STOP_CAPTURE,
119     EVENT_AUDIO_ENDED,
120     EVENT_RECOGNITION_ENDED,
121     EVENT_MAX_VALUE = EVENT_RECOGNITION_ENDED
122   };
123 
124   struct Session {
125     Session();
126     ~Session();
127 
128     int id;
129     bool abort_requested;
130     bool listener_is_active;
131     SpeechRecognitionSessionConfig config;
132     SpeechRecognitionSessionContext context;
133     scoped_refptr<SpeechRecognizer> recognizer;
134     scoped_ptr<MediaStreamUIProxy> ui;
135   };
136 
137   // Callback issued by the SpeechRecognitionManagerDelegate for reporting
138   // asynchronously the result of the CheckRecognitionIsAllowed call.
139   void RecognitionAllowedCallback(int session_id,
140                                   bool ask_user,
141                                   bool is_allowed);
142 
143   // Callback to get back the result of a media request. |devices| is an array
144   // of devices approved to be used for the request, |devices| is empty if the
145   // users deny the request.
146   void MediaRequestPermissionCallback(int session_id,
147                                       const MediaStreamDevices& devices,
148                                       scoped_ptr<MediaStreamUIProxy> stream_ui);
149 
150   // Entry point for pushing any external event into the session handling FSM.
151   void DispatchEvent(int session_id, FSMEvent event);
152 
153   // Defines the behavior of the session handling FSM, selecting the appropriate
154   // transition according to the session, its current state and the event.
155   void ExecuteTransitionAndGetNextState(Session* session,
156                                         FSMState session_state,
157                                         FSMEvent event);
158 
159   // Retrieves the state of the session, enquiring directly the recognizer.
160   FSMState GetSessionState(int session_id) const;
161 
162   // The methods below handle transitions of the session handling FSM.
163   void SessionStart(const Session& session);
164   void SessionAbort(const Session& session);
165   void SessionStopAudioCapture(const Session& session);
166   void ResetCapturingSessionId(const Session& session);
167   void SessionDelete(Session* session);
168   void NotFeasible(const Session& session, FSMEvent event);
169 
170   bool SessionExists(int session_id) const;
171   Session* GetSession(int session_id) const;
172   SpeechRecognitionEventListener* GetListener(int session_id) const;
173   SpeechRecognitionEventListener* GetDelegateListener() const;
174   int GetNextSessionID();
175 
176   media::AudioManager* audio_manager_;
177   MediaStreamManager* media_stream_manager_;
178   typedef std::map<int, Session*> SessionsTable;
179   SessionsTable sessions_;
180   int primary_session_id_;
181   int last_session_id_;
182   bool is_dispatching_event_;
183   scoped_ptr<SpeechRecognitionManagerDelegate> delegate_;
184 
185   // Used for posting asynchronous tasks (on the IO thread) without worrying
186   // about this class being destroyed in the meanwhile (due to browser shutdown)
187   // since tasks pending on a destroyed WeakPtr are automatically discarded.
188   base::WeakPtrFactory<SpeechRecognitionManagerImpl> weak_factory_;
189 };
190 
191 }  // namespace content
192 
193 #endif  // CONTENT_BROWSER_SPEECH_SPEECH_RECOGNITION_MANAGER_IMPL_H_
194