summaryrefslogtreecommitdiffstats
path: root/media/audio/linux/alsa_output.h
blob: b6d405b3fd90dbd715a03fd16a3df2df147a3efe (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
// Copyright (c) 2009 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
//
// Creates an output stream based on the ALSA PCM interface.  The current
// implementation creates one thread per ALSA playback handle that is
// responsible for synchronously pulling data from the audio data source.
//
// This output stream buffers in two places:
//   (1) In the ALSA device
//   (2) In an in-memory buffer.
//
// The ALSA device buffer is kept as full as possible.  The in-memory buffer
// attempts to keep enough extra data so that |min_buffer_ms| worth of data
// is available between the in-memory buffer and the device buffer.  Requests
// to the audio data source are made if the total amount buffered falls below
// |min_buffer_ms|.
//
// On device write failure, the stream will move into an invalid state.  No
// more data will be pulled from the data source, and the playback thread will
// be stopped.
//
// If the stream is successfully opened, Close() must be called before the
// stream is deleted.

#ifndef MEDIA_AUDIO_LINUX_ALSA_OUTPUT_H_
#define MEDIA_AUDIO_LINUX_ALSA_OUTPUT_H_

#include <alsa/asoundlib.h>

#include <deque>
#include <string>

#include "base/lock.h"
#include "base/ref_counted.h"
#include "base/scoped_ptr.h"
#include "base/thread.h"
#include "media/audio/audio_output.h"

class Thread;

class AlsaPCMOutputStream :
    public AudioOutputStream,
    public base::RefCountedThreadSafe<AlsaPCMOutputStream> {
 public:
  // Set to "plug:default" which should allow ALSA to do whatever is
  // necessary to simulate an audio device capable of handling our PCMs.
  static const char* kDefaultDevice;

  // Create a PCM Output stream for the ALSA device identified by
  // |device_name|.  If unsure of hte device_name, use kDefaultDevice.
  AlsaPCMOutputStream(const std::string& device_name,
                      int min_buffer_ms,
                      AudioManager::Format format,
                      int channels,
                      int sample_rate,
                      char bits_per_sample);
  virtual ~AlsaPCMOutputStream();

  // Implementation of AudioOutputStream.
  virtual bool Open(size_t packet_size);
  virtual void Close();
  virtual void Start(AudioSourceCallback* callback);
  virtual void Stop();
  virtual void SetVolume(double left_level, double right_level);
  virtual void GetVolume(double* left_level, double* right_level);
  virtual size_t GetNumBuffers();

 private:
  // Closes the playback handle, reporting errors if any occur.  Returns true
  // if the device was successfully closed.
  bool CloseDevice_Locked();

  // Stops playback, ignoring state checks.
  void StopInternal_Locked();

  // Moves the stream into the error state, setting the correct internal flags.
  // Ensure that all resources are cleaned up before executing this function.
  void EnterStateError_Locked();

  // Releases all the resources in the audio stream.  This method will also
  // terminate the playback thread itself.
  //
  // This method must be run in the |playback_thead_|.
  void ReleaseResources();

  // Retrieve the total number of frames buffered in both memory and in the
  // audio device.  Use this to determine if more data should be requested from
  // the audio source.
  snd_pcm_sframes_t GetFramesOfDelay_Locked();

  // Buffer more packets from data source if necessary.
  //
  // This function must be run in the |playback_thread_|.
  void BufferPackets();

  // Returns true if our buffer is underfull.
  bool ShouldBufferMore_NoLock();

  // Write as many buffered packets into the device as there is room in the
  // device buffer.
  //
  // This function must be run in the |playback_thread_|.
  void FillAlsaDeviceBuffer();

  // State of the stream.
  State state_;

  // The ALSA device name to use.
  std::string device_name_;

  // Handle to the actual PCM playback device.
  snd_pcm_t* playback_handle_;

  // Period size for ALSA ring-buffer.  Basically, how long to wait between
  // writes.
  snd_pcm_sframes_t period_size_;

  // Callback used to request more data from the data source.
  AudioSourceCallback* source_callback_;

  // Playback thread.
  base::Thread playback_thread_;

  // Lock for field access to this object.
  Lock lock_;

  // Sample format configuration.
  snd_pcm_format_t pcm_format_;
  const int channels_;
  const int sample_rate_;
  const char bits_per_sample_;
  char bytes_per_frame_;

  // In-memory buffer to hold sound samples before pushing to the sound device.
  // TODO(ajwong): There are now about 3 buffer queue implementations. Factor
  // them out.
  struct Packet {
    explicit Packet(int new_capacity)
        : capacity(new_capacity),
          size(0),
          used(0),
          buffer(new char[capacity]) {
    }
    size_t capacity;
    size_t size;
    size_t used;
    scoped_array<char> buffer;
  };
  int min_buffer_frames_;
  std::deque<Packet*> buffered_packets_;
  size_t packet_size_;

  // Flag indiciating the device write tasks have stopped scheduling
  // themselves.  This should only be modified by the BufferPackets() and
  // FillAlsaDeviceBuffer() methods.
  bool device_write_suspended_;

  // Flag indicating that the resources are already cleaned.
  bool resources_released_;

  DISALLOW_COPY_AND_ASSIGN(AlsaPCMOutputStream);
};

#endif  // MEDIA_AUDIO_LINUX_ALSA_OUTPUT_H_