summaryrefslogtreecommitdiffstats
path: root/third_party/libwebrtc/modules/audio_device/opensl
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/libwebrtc/modules/audio_device/opensl')
-rw-r--r--third_party/libwebrtc/modules/audio_device/opensl/single_rw_fifo.cc123
-rw-r--r--third_party/libwebrtc/modules/audio_device/opensl/single_rw_fifo.h45
2 files changed, 168 insertions, 0 deletions
diff --git a/third_party/libwebrtc/modules/audio_device/opensl/single_rw_fifo.cc b/third_party/libwebrtc/modules/audio_device/opensl/single_rw_fifo.cc
new file mode 100644
index 0000000000..a7bad3af6e
--- /dev/null
+++ b/third_party/libwebrtc/modules/audio_device/opensl/single_rw_fifo.cc
@@ -0,0 +1,123 @@
+/*
+ * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#if defined(_MSC_VER)
+#include <windows.h>
+#endif
+
+#include "single_rw_fifo.h"
+
+#include <assert.h>
+
+static int UpdatePos(int pos, int capacity) {
+ return (pos + 1) % capacity;
+}
+
+namespace webrtc {
+
+namespace subtle {
+
+// Start with compiler support, then processor-specific hacks
+#if defined(__GNUC__) || defined(__clang__)
+// Available on GCC and clang - others?
+inline void MemoryBarrier() {
+ __sync_synchronize();
+}
+
+#elif defined(_MSC_VER)
+inline void MemoryBarrier() {
+ ::MemoryBarrier();
+}
+
+#elif defined(__aarch64__)
+// From http://http://src.chromium.org/viewvc/chrome/trunk/src/base/atomicops_internals_arm64_gcc.h
+inline void MemoryBarrier() {
+ __asm__ __volatile__ ("dmb ish" ::: "memory");
+}
+
+#elif defined(__ARMEL__)
+// From http://src.chromium.org/viewvc/chrome/trunk/src/base/atomicops_internals_arm_gcc.h
+inline void MemoryBarrier() {
+ // Note: This is a function call, which is also an implicit compiler barrier.
+ typedef void (*KernelMemoryBarrierFunc)();
+ ((KernelMemoryBarrierFunc)0xffff0fa0)();
+}
+
+#elif defined(__x86_64__) || defined (__i386__)
+// From http://src.chromium.org/viewvc/chrome/trunk/src/base/atomicops_internals_x86_gcc.h
+// mfence exists on x64 and x86 platforms containing SSE2.
+// x86 platforms that don't have SSE2 will crash with SIGILL.
+// If this code needs to run on such platforms in the future,
+// add runtime CPU detection here.
+inline void MemoryBarrier() {
+ __asm__ __volatile__("mfence" : : : "memory");
+}
+
+#elif defined(__MIPSEL__)
+// From http://src.chromium.org/viewvc/chrome/trunk/src/base/atomicops_internals_mips_gcc.h
+inline void MemoryBarrier() {
+ __asm__ __volatile__("sync" : : : "memory");
+}
+
+#else
+#error Add an implementation of MemoryBarrier() for this platform!
+#endif
+
+} // namespace subtle
+
+SingleRwFifo::SingleRwFifo(int capacity)
+ : queue_(new int8_t*[capacity]),
+ capacity_(capacity),
+ size_(0),
+ read_pos_(0),
+ write_pos_(0)
+{
+}
+
+SingleRwFifo::~SingleRwFifo() {
+}
+
+void SingleRwFifo::Push(int8_t* mem) {
+ assert(mem);
+
+ // Ensure that there is space for the new data in the FIFO.
+ // Note there is only one writer meaning that the other thread is guaranteed
+ // only to decrease the size.
+ const int free_slots = capacity() - size();
+ if (free_slots <= 0) {
+ // Size can be queried outside of the Push function. The caller is assumed
+ // to ensure that Push will be successful before calling it.
+ assert(false);
+ return;
+ }
+ queue_.get()[write_pos_] = mem;
+ // Memory barrier ensures that |size_| is updated after the size has changed.
+ subtle::MemoryBarrier();
+ ++size_;
+ write_pos_ = UpdatePos(write_pos_, capacity());
+}
+
+int8_t* SingleRwFifo::Pop() {
+ int8_t* ret_val = NULL;
+ if (size() <= 0) {
+ // Size can be queried outside of the Pop function. The caller is assumed
+ // to ensure that Pop will be successfull before calling it.
+ assert(false);
+ return ret_val;
+ }
+ ret_val = queue_.get()[read_pos_];
+ // Memory barrier ensures that |size_| is updated after the size has changed.
+ subtle::MemoryBarrier();
+ --size_;
+ read_pos_ = UpdatePos(read_pos_, capacity());
+ return ret_val;
+}
+
+} // namespace webrtc
diff --git a/third_party/libwebrtc/modules/audio_device/opensl/single_rw_fifo.h b/third_party/libwebrtc/modules/audio_device/opensl/single_rw_fifo.h
new file mode 100644
index 0000000000..0db4815fa5
--- /dev/null
+++ b/third_party/libwebrtc/modules/audio_device/opensl/single_rw_fifo.h
@@ -0,0 +1,45 @@
+/*
+ * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#ifndef WEBRTC_MODULES_AUDIO_DEVICE_ANDROID_SINGLE_RW_FIFO_H_
+#define WEBRTC_MODULES_AUDIO_DEVICE_ANDROID_SINGLE_RW_FIFO_H_
+
+namespace webrtc {
+
+// Implements a lock-free FIFO losely based on
+// http://src.chromium.org/viewvc/chrome/trunk/src/media/base/audio_fifo.cc
+// Note that this class assumes there is one producer (writer) and one
+// consumer (reader) thread.
+class SingleRwFifo {
+ public:
+ explicit SingleRwFifo(int capacity);
+ ~SingleRwFifo();
+
+ void Push(int8_t* mem);
+ int8_t* Pop();
+
+ void Clear();
+
+ int size() { return size_; }
+ int capacity() const { return capacity_; }
+
+ private:
+ std::unique_ptr<int8_t* []> queue_;
+ int capacity_;
+
+ std::atomic<int32_t> size_;
+
+ int read_pos_;
+ int write_pos_;
+};
+
+} // namespace webrtc
+
+#endif // WEBRTC_MODULES_AUDIO_DEVICE_ANDROID_SINGLE_RW_FIFO_H_