Refactor for AudioTrack/Record callback interface

Replace libaudioclient callback functions with appropriate
interfaces. Control callback object lifetime with ref-counting.
Misc cleanup including using sp<> where appropriate.

Test: OboeTester Output/Input streams
Bug: 199156212
Bug: 216175830
Change-Id: I366c543e85a62f878908836e9ad1914182dc9e6f
diff --git a/src/android/AudioPlayer_to_android.cpp b/src/android/AudioPlayer_to_android.cpp
index a324b4a..ef466b6 100644
--- a/src/android/AudioPlayer_to_android.cpp
+++ b/src/android/AudioPlayer_to_android.cpp
@@ -19,6 +19,7 @@
 #include "android/android_AudioToCbRenderer.h"
 #include "android/android_StreamPlayer.h"
 #include "android/android_LocAVPlayer.h"
+#include "android/AudioTrackCallback.h"
 #include "android/include/AacBqToPcmCbRenderer.h"
 #include "android/channels.h"
 
@@ -1233,23 +1234,15 @@
 //-----------------------------------------------------------------------------
 // Callback associated with an AudioTrack of an SL ES AudioPlayer that gets its data
 // from a buffer queue. This will not be called once the AudioTrack has been destroyed.
-static void audioTrack_callBack_pullFromBuffQueue(int event, void* user, void *info) {
-    CAudioPlayer *ap = (CAudioPlayer *)user;
-
-    if (!android::CallbackProtector::enterCbIfOk(ap->mCallbackProtector)) {
-        // it is not safe to enter the callback (the track is about to go away)
-        return;
-    }
+size_t audioTrack_handleMoreData_lockPlay(CAudioPlayer* ap,
+                                        const android::AudioTrack::Buffer& buffer) {
 
     void * callbackPContext = NULL;
-    switch (event) {
-
-    case android::AudioTrack::EVENT_MORE_DATA: {
+    size_t bytesWritten = 0;
         //SL_LOGV("received event EVENT_MORE_DATA from AudioTrack TID=%d", gettid());
         slPrefetchCallback prefetchCallback = NULL;
         void *prefetchContext = NULL;
         SLuint32 prefetchEvents = SL_PREFETCHEVENT_NONE;
-        android::AudioTrack::Buffer* pBuff = (android::AudioTrack::Buffer*)info;
 
         // retrieve data from the buffer queue
         interface_lock_exclusive(&ap->mBufferQueue);
@@ -1274,17 +1267,15 @@
             BufferHeader *newFront = &oldFront[1];
 
             size_t availSource = oldFront->mSize - ap->mBufferQueue.mSizeConsumed;
-            size_t availSink = pBuff->size;
+            size_t availSink = buffer.size();
             size_t bytesToCopy = availSource < availSink ? availSource : availSink;
             void *pSrc = (char *)oldFront->mBuffer + ap->mBufferQueue.mSizeConsumed;
-            memcpy(pBuff->raw, pSrc, bytesToCopy);
-
+            memcpy(buffer.data(), pSrc, bytesToCopy);
+            bytesWritten = bytesToCopy;
             if (bytesToCopy < availSource) {
                 ap->mBufferQueue.mSizeConsumed += bytesToCopy;
-                // pBuff->size is already equal to bytesToCopy in this case
             } else {
                 // consumed an entire buffer, dequeue
-                pBuff->size = bytesToCopy;
                 ap->mBufferQueue.mSizeConsumed = 0;
                 if (newFront ==
                         &ap->mBufferQueue.mArray
@@ -1299,8 +1290,6 @@
                 ap->mBufferQueue.mCallbackPending = true;
             }
         } else { // empty queue
-            // signal no data available
-            pBuff->size = 0;
 
             // signal we're at the end of the content, but don't pause (see note in function)
             audioPlayer_dispatch_headAtEnd_lockPlay(ap, false /*set state to paused?*/, false);
@@ -1336,41 +1325,8 @@
                         SL_PREFETCHEVENT_FILLLEVELCHANGE);
             }
         }
-    }
-    break;
 
-    case android::AudioTrack::EVENT_MARKER:
-        //SL_LOGI("received event EVENT_MARKER from AudioTrack");
-        audioTrack_handleMarker_lockPlay(ap);
-        break;
-
-    case android::AudioTrack::EVENT_NEW_POS:
-        //SL_LOGI("received event EVENT_NEW_POS from AudioTrack");
-        audioTrack_handleNewPos_lockPlay(ap);
-        break;
-
-    case android::AudioTrack::EVENT_UNDERRUN:
-        //SL_LOGI("received event EVENT_UNDERRUN from AudioTrack");
-        audioTrack_handleUnderrun_lockPlay(ap);
-        break;
-
-    case android::AudioTrack::EVENT_NEW_IAUDIOTRACK:
-        // ignore for now
-        break;
-
-    case android::AudioTrack::EVENT_BUFFER_END:
-    case android::AudioTrack::EVENT_LOOP_END:
-    case android::AudioTrack::EVENT_STREAM_END:
-        // These are unexpected so fall through
-        FALLTHROUGH_INTENDED;
-    default:
-        // FIXME where does the notification of SL_PLAYEVENT_HEADMOVING fit?
-        SL_LOGE("Encountered unknown AudioTrack event %d for CAudioPlayer %p", event,
-                (CAudioPlayer *)user);
-        break;
-    }
-
-    ap->mCallbackProtector->exitCb();
+    return bytesWritten;
 }
 
 
@@ -1684,16 +1640,15 @@
         } else {
             notificationFrames = 0;
         }
-
-        android::AudioTrack* pat = new android::AudioTrack(
+        const auto callbackHandle = android::sp<android::AudioTrackCallback>::make(pAudioPlayer);
+        const auto pat = android::sp<android::AudioTrack>::make(
                 pAudioPlayer->mStreamType,                           // streamType
                 sampleRate,                                          // sampleRate
                 sles_to_android_sampleFormat(df_pcm),                // format
                 channelMask,                                         // channel mask
                 0,                                                   // frameCount
                 policy,                                              // flags
-                audioTrack_callBack_pullFromBuffQueue,               // callback
-                (void *) pAudioPlayer,                               // user
+                callbackHandle,                                      // callback
                 notificationFrames,                                  // see comment above
                 pAudioPlayer->mSessionId);
 
@@ -1702,17 +1657,17 @@
 
         android::status_t status = pat->initCheck();
         if (status != android::NO_ERROR) {
-            // AudioTracks are meant to be refcounted, so their dtor is protected.
-            static_cast<void>(android::sp<android::AudioTrack>(pat));
-
             SL_LOGE("AudioTrack::initCheck status %u", status);
             // FIXME should return a more specific result depending on status
             result = SL_RESULT_CONTENT_UNSUPPORTED;
             return result;
         }
 
-        pAudioPlayer->mTrackPlayer->init(pat, android::PLAYER_TYPE_SLES_AUDIOPLAYER_BUFFERQUEUE,
-                usageForStreamType(pAudioPlayer->mStreamType), pAudioPlayer->mSessionId);
+        pAudioPlayer->mTrackPlayer->init(
+            pat, callbackHandle,
+            android::PLAYER_TYPE_SLES_AUDIOPLAYER_BUFFERQUEUE,
+            usageForStreamType(pAudioPlayer->mStreamType),
+            pAudioPlayer->mSessionId);
 
         // update performance mode according to actual flags granted to AudioTrack
         checkAndSetPerformanceModePost(pAudioPlayer);
diff --git a/src/android/AudioRecordCallback.h b/src/android/AudioRecordCallback.h
new file mode 100644
index 0000000..53b7e35
--- /dev/null
+++ b/src/android/AudioRecordCallback.h
@@ -0,0 +1,80 @@
+/*
+ * Copyright (C) 2022 The Android Open Source Project
+ *
+ * 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.
+ */
+
+#pragma once
+
+#ifndef SL_PREFETCHEVENT_NONE // This is defined in slesl_allinclusive, which isn't guarded
+#include "sles_allinclusive.h"
+#endif
+
+#include "media/AudioRecord.h"
+
+void audioRecorder_handleOverrun_lockRecord(CAudioRecorder* ar);
+void audioRecorder_handleNewPos_lockRecord(CAudioRecorder* ar);
+void audioRecorder_handleMarker_lockRecord(CAudioRecorder* ar);
+size_t audioRecorder_handleMoreData_lockRecord(CAudioRecorder* ar,
+                                               const android::AudioRecord::Buffer&);
+//--------------------------------------------------------------------------------------------------
+namespace android {
+
+class AudioRecordCallback : public android::AudioRecord::IAudioRecordCallback {
+  public:
+    AudioRecordCallback(CAudioRecorder * audioRecorder) : mAr(audioRecorder) {}
+    AudioRecordCallback(const AudioRecordCallback&) = delete;
+    AudioRecordCallback& operator=(const AudioRecordCallback&) = delete;
+
+  private:
+    size_t onMoreData(const android::AudioRecord::Buffer& buffer) override {
+        if (!android::CallbackProtector::enterCbIfOk(mAr->mCallbackProtector)) {
+            // it is not safe to enter the callback (the track is about to go away)
+            return buffer.size(); // replicate existing behavior
+        }
+        size_t bytesRead = audioRecorder_handleMoreData_lockRecord(mAr, buffer);
+        mAr->mCallbackProtector->exitCb();
+        return bytesRead;
+    }
+
+
+    void onOverrun() override {
+        if (!android::CallbackProtector::enterCbIfOk(mAr->mCallbackProtector)) {
+            // it is not safe to enter the callback (the track is about to go away)
+            return;
+        }
+        audioRecorder_handleOverrun_lockRecord(mAr);
+        mAr->mCallbackProtector->exitCb();
+    }
+    void onMarker(uint32_t) override {
+        if (!android::CallbackProtector::enterCbIfOk(mAr->mCallbackProtector)) {
+            // it is not safe to enter the callback (the track is about to go away)
+            return;
+        }
+
+        audioRecorder_handleMarker_lockRecord(mAr);
+        mAr->mCallbackProtector->exitCb();
+    }
+    void onNewPos(uint32_t) override {
+        if (!android::CallbackProtector::enterCbIfOk(mAr->mCallbackProtector)) {
+            // it is not safe to enter the callback (the track is about to go away)
+            return;
+        }
+
+        audioRecorder_handleNewPos_lockRecord(mAr);
+        mAr->mCallbackProtector->exitCb();
+    }
+    CAudioRecorder * const mAr;
+};
+
+} // namespace android
diff --git a/src/android/AudioRecorder_to_android.cpp b/src/android/AudioRecorder_to_android.cpp
index 411beff..d4b2964 100644
--- a/src/android/AudioRecorder_to_android.cpp
+++ b/src/android/AudioRecorder_to_android.cpp
@@ -24,6 +24,7 @@
 #include <SLES/OpenSLES_Android.h>
 
 #include <android_runtime/AndroidRuntime.h>
+#include <android/AudioRecordCallback.h>
 
 #define KEY_RECORDING_SOURCE_PARAMSIZE  sizeof(SLuint32)
 #define KEY_RECORDING_PRESET_PARAMSIZE  sizeof(SLuint32)
@@ -307,96 +308,58 @@
     return SL_RESULT_SUCCESS;
 }
 //-----------------------------------------------------------------------------
-static void audioRecorder_callback(int event, void* user, void *info) {
+size_t audioRecorder_handleMoreData_lockRecord(CAudioRecorder* ar,
+                                               const android::AudioRecord::Buffer& buffer) {
     //SL_LOGV("audioRecorder_callback(%d, %p, %p) entering", event, user, info);
 
-    CAudioRecorder *ar = (CAudioRecorder *)user;
-
-    if (!android::CallbackProtector::enterCbIfOk(ar->mCallbackProtector)) {
-        // it is not safe to enter the callback (the track is about to go away)
-        return;
-    }
-
     void * callbackPContext = NULL;
+    size_t bytesRead = 0;
+    slBufferQueueCallback callback = NULL;
 
-    switch (event) {
-    case android::AudioRecord::EVENT_MORE_DATA: {
-        slBufferQueueCallback callback = NULL;
-        android::AudioRecord::Buffer* pBuff = (android::AudioRecord::Buffer*)info;
+    // push data to the buffer queue
+    interface_lock_exclusive(&ar->mBufferQueue);
 
-        // push data to the buffer queue
-        interface_lock_exclusive(&ar->mBufferQueue);
+    if (ar->mBufferQueue.mState.count != 0) {
+        assert(ar->mBufferQueue.mFront != ar->mBufferQueue.mRear);
 
-        if (ar->mBufferQueue.mState.count != 0) {
-            assert(ar->mBufferQueue.mFront != ar->mBufferQueue.mRear);
+        BufferHeader *oldFront = ar->mBufferQueue.mFront;
+        BufferHeader *newFront = &oldFront[1];
 
-            BufferHeader *oldFront = ar->mBufferQueue.mFront;
-            BufferHeader *newFront = &oldFront[1];
-
-            size_t availSink = oldFront->mSize - ar->mBufferQueue.mSizeConsumed;
-            size_t availSource = pBuff->size;
-            size_t bytesToCopy = availSink < availSource ? availSink : availSource;
-            void *pDest = (char *)oldFront->mBuffer + ar->mBufferQueue.mSizeConsumed;
-            memcpy(pDest, pBuff->raw, bytesToCopy);
-
-            if (bytesToCopy < availSink) {
-                // can't consume the whole or rest of the buffer in one shot
-                ar->mBufferQueue.mSizeConsumed += availSource;
-                // pBuff->size is already equal to bytesToCopy in this case
-            } else {
-                // finish pushing the buffer or push the buffer in one shot
-                pBuff->size = bytesToCopy;
-                ar->mBufferQueue.mSizeConsumed = 0;
-                if (newFront == &ar->mBufferQueue.mArray[ar->mBufferQueue.mNumBuffers + 1]) {
-                    newFront = ar->mBufferQueue.mArray;
-                }
-                ar->mBufferQueue.mFront = newFront;
-
-                ar->mBufferQueue.mState.count--;
-                ar->mBufferQueue.mState.playIndex++;
-
-                // data has been copied to the buffer, and the buffer queue state has been updated
-                // we will notify the client if applicable
-                callback = ar->mBufferQueue.mCallback;
-                // save callback data
-                callbackPContext = ar->mBufferQueue.mContext;
+        size_t availSink = oldFront->mSize - ar->mBufferQueue.mSizeConsumed;
+        size_t availSource = buffer.size();
+        size_t bytesToCopy = availSink < availSource ? availSink : availSource;
+        void *pDest = (char *)oldFront->mBuffer + ar->mBufferQueue.mSizeConsumed;
+        memcpy(pDest, buffer.data(), bytesToCopy);
+        bytesRead = bytesToCopy;
+        if (bytesToCopy < availSink) {
+            // can't consume the whole or rest of the buffer in one shot
+            ar->mBufferQueue.mSizeConsumed += availSource;
+        } else {
+            // finish pushing the buffer or push the buffer in one shot
+            ar->mBufferQueue.mSizeConsumed = 0;
+            if (newFront == &ar->mBufferQueue.mArray[ar->mBufferQueue.mNumBuffers + 1]) {
+                newFront = ar->mBufferQueue.mArray;
             }
-        } else { // empty queue
-            // no destination to push the data
-            pBuff->size = 0;
+            ar->mBufferQueue.mFront = newFront;
+
+            ar->mBufferQueue.mState.count--;
+            ar->mBufferQueue.mState.playIndex++;
+
+            // data has been copied to the buffer, and the buffer queue state has been updated
+            // we will notify the client if applicable
+            callback = ar->mBufferQueue.mCallback;
+            // save callback data
+            callbackPContext = ar->mBufferQueue.mContext;
         }
-
-        interface_unlock_exclusive(&ar->mBufferQueue);
-
-        // notify client
-        if (NULL != callback) {
-            (*callback)(&ar->mBufferQueue.mItf, callbackPContext);
-        }
-        }
-        break;
-
-    case android::AudioRecord::EVENT_OVERRUN:
-        audioRecorder_handleOverrun_lockRecord(ar);
-        break;
-
-    case android::AudioRecord::EVENT_MARKER:
-        audioRecorder_handleMarker_lockRecord(ar);
-        break;
-
-    case android::AudioRecord::EVENT_NEW_POS:
-        audioRecorder_handleNewPos_lockRecord(ar);
-        break;
-
-    case android::AudioRecord::EVENT_NEW_IAUDIORECORD:
-        // ignore for now
-        break;
-
-    default:
-        SL_LOGE("Encountered unknown AudioRecord event %d for CAudioRecord %p", event, ar);
-        break;
     }
 
-    ar->mCallbackProtector->exitCb();
+    interface_unlock_exclusive(&ar->mBufferQueue);
+
+    // notify client
+    if (NULL != callback) {
+        (*callback)(&ar->mBufferQueue.mItf, callbackPContext);
+    }
+    return bytesRead;
 }
 
 
@@ -420,6 +383,7 @@
         // microphone to simple buffer queue
         ar->mAndroidObjType = AUDIORECORDER_FROM_MIC_TO_PCM_BUFFERQUEUE;
         ar->mAudioRecord.clear();
+        ar->mCallbackHandle.clear();
         ar->mCallbackProtector = new android::CallbackProtector();
         ar->mRecordSource = AUDIO_SOURCE_DEFAULT;
         ar->mPerformanceMode = ANDROID_PERFORMANCE_MODE_DEFAULT;
@@ -694,7 +658,7 @@
     attributionSource.uid = VALUE_OR_FATAL(android::legacy2aidl_uid_t_int32_t(getuid()));
     attributionSource.pid = VALUE_OR_FATAL(android::legacy2aidl_pid_t_int32_t(getpid()));
     attributionSource.token = android::sp<android::BBinder>::make();
-
+    ar->mCallbackHandle = android::sp<android::AudioRecordCallback>::make(ar);
     // initialize platform-specific CAudioRecorder fields
     ar->mAudioRecord = new android::AudioRecord(
             ar->mRecordSource,     // source
@@ -703,8 +667,7 @@
             channelMask,           // channel mask
             attributionSource,
             0,                     // frameCount
-            audioRecorder_callback,// callback_t
-            (void*)ar,             // user, callback data, here the AudioRecorder
+            ar->mCallbackHandle,
             0,                     // notificationFrames
             AUDIO_SESSION_ALLOCATE,
             android::AudioRecord::TRANSFER_CALLBACK,
@@ -721,6 +684,7 @@
         // FIXME should return a more specific result depending on status
         result = SL_RESULT_CONTENT_UNSUPPORTED;
         ar->mAudioRecord.clear();
+        ar->mCallbackHandle.clear();
         return result;
     }
 
@@ -744,6 +708,7 @@
             SL_LOGE("Java exception releasing recorder routing object.");
             result = SL_RESULT_INTERNAL_ERROR;
             ar->mAudioRecord.clear();
+            ar->mCallbackHandle.clear();
             return result;
         }
    }
@@ -810,6 +775,7 @@
         ar->mAudioRecord.clear();
     }
     // explicit destructor
+    ar->mCallbackHandle.~sp();
     ar->mAudioRecord.~sp();
     ar->mCallbackProtector.~sp();
 }
diff --git a/src/android/AudioTrackCallback.h b/src/android/AudioTrackCallback.h
new file mode 100644
index 0000000..7020606
--- /dev/null
+++ b/src/android/AudioTrackCallback.h
@@ -0,0 +1,98 @@
+/*
+ * Copyright (C) 2022 The Android Open Source Project
+ *
+ * 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.
+ */
+
+#pragma once
+
+#ifndef SL_PREFETCHEVENT_NONE // This is defined in slesl_allinclusive, which isn't guarded
+#include "sles_allinclusive.h"
+#endif
+
+#include "media/AudioTrack.h"
+
+void audioPlayer_dispatch_headAtEnd_lockPlay(CAudioPlayer*, bool, bool);
+
+void audioTrack_handleUnderrun_lockPlay(CAudioPlayer* ap);
+void audioTrack_handleMarker_lockPlay(CAudioPlayer* ap);
+void audioTrack_handleNewPos_lockPlay(CAudioPlayer* ap);
+size_t audioTrack_handleMoreData_lockPlay(CAudioPlayer* ap,
+                                        const android::AudioTrack::Buffer& buffer);
+//--------------------------------------------------------------------------------------------------
+namespace android {
+class AudioTrackCallback : public AudioTrack::IAudioTrackCallback {
+  public:
+    AudioTrackCallback(CAudioPlayer * player) : mAp(player) {}
+
+    size_t onMoreData(const AudioTrack::Buffer& buffer) override {
+        if (!android::CallbackProtector::enterCbIfOk(mAp->mCallbackProtector)) {
+          // it is not safe to enter the callback (the track is about to go away)
+          return buffer.size(); // duplicate existing behavior
+        }
+        size_t bytesCopied = audioTrack_handleMoreData_lockPlay(mAp, buffer);
+        mAp->mCallbackProtector->exitCb();
+        return bytesCopied;
+      }
+
+    void onUnderrun() override {
+        if (!android::CallbackProtector::enterCbIfOk(mAp->mCallbackProtector)) {
+          // it is not safe to enter the callback (the track is about to go away)
+            return;
+        }
+        audioTrack_handleUnderrun_lockPlay(mAp);
+        mAp->mCallbackProtector->exitCb();
+    }
+
+    void onLoopEnd([[maybe_unused]] int32_t loopsRemaining) override {
+        SL_LOGE("Encountered loop end for CAudioPlayer %p", mAp);
+    }
+    void onMarker([[maybe_unused]] uint32_t markerPosition) override {
+        if (!android::CallbackProtector::enterCbIfOk(mAp->mCallbackProtector)) {
+          // it is not safe to enter the callback (the track is about to go away)
+          return;
+        }
+        audioTrack_handleMarker_lockPlay(mAp);
+        mAp->mCallbackProtector->exitCb();
+    }
+
+    void onNewPos([[maybe_unused]] uint32_t newPos) override {
+        if (!android::CallbackProtector::enterCbIfOk(mAp->mCallbackProtector)) {
+          // it is not safe to enter the callback (the track is about to go away)
+          return;
+        }
+        audioTrack_handleNewPos_lockPlay(mAp);
+        mAp->mCallbackProtector->exitCb();
+    }
+    void onBufferEnd() override {
+        SL_LOGE("Encountered buffer end for CAudioPlayer %p", mAp);
+    }
+    // Ignore
+    void onNewIAudioTrack() override {}
+    void onStreamEnd() override {
+        SL_LOGE("Encountered buffer end for CAudioPlayer %p", mAp);
+    }
+    void onNewTimestamp([[maybe_unused]] AudioTimestamp timestamp) {
+        SL_LOGE("Encountered write more data for CAudioPlayer %p", mAp);
+    }
+    size_t onCanWriteMoreData([[maybe_unused]] const AudioTrack::Buffer& buffer) {
+        SL_LOGE("Encountered write more data for CAudioPlayer %p", mAp);
+        return 0;
+    }
+
+  private:
+    AudioTrackCallback(const AudioTrackCallback&) = delete;
+    AudioTrackCallback& operator=(const AudioTrackCallback&) = delete;
+    CAudioPlayer* const mAp;
+};
+}  // namespace android
diff --git a/src/classes.h b/src/classes.h
index fc64f25..c42fde4 100644
--- a/src/classes.h
+++ b/src/classes.h
@@ -21,6 +21,7 @@
 #include "android/android_GenericPlayer.h"
 #include <media/TrackPlayerBase.h>
 #include <audiomanager/IAudioManager.h>
+namespace android { class AudioRecordCallback; };
 #endif
 
 // Class structures
@@ -161,6 +162,7 @@
     enum AndroidObjectType mAndroidObjType;
     android::sp<android::AudioRecord> mAudioRecord;
     android::sp<android::CallbackProtector> mCallbackProtector;
+    android::sp<android::AudioRecordCallback> mCallbackHandle;
     audio_source_t mRecordSource;
     SLuint32 mPerformanceMode;
 #endif
diff --git a/src/itf/IEngine.cpp b/src/itf/IEngine.cpp
index da5623c..be2d2b0 100644
--- a/src/itf/IEngine.cpp
+++ b/src/itf/IEngine.cpp
@@ -459,6 +459,7 @@
                     // FIXME unnecessary once those fields are encapsulated in one class, rather
                     //   than a structure
                     (void) new (&thiz->mAudioRecord) android::sp<android::AudioRecord>();
+                    (void) new (&thiz->mCallbackHandle) android::sp<android::AudioRecordCallback>();
                     (void) new (&thiz->mCallbackProtector)
                             android::sp<android::CallbackProtector>();
                     thiz->mRecordSource = AUDIO_SOURCE_DEFAULT;