Skip to content

Commit

Permalink
Merge "Use sp<LooperCallback> to fix race causing dangling pointer." …
Browse files Browse the repository at this point in the history
…into jb-dev
  • Loading branch information
Jeff Brown authored and Android (Google) Code Review committed Jun 1, 2012
2 parents 53913ed + 80a1de1 commit 6e8e41a
Show file tree
Hide file tree
Showing 2 changed files with 40 additions and 37 deletions.
42 changes: 22 additions & 20 deletions core/jni/android_view_DisplayEventReceiver.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -42,12 +42,13 @@ static struct {
} gDisplayEventReceiverClassInfo;


class NativeDisplayEventReceiver : public RefBase {
class NativeDisplayEventReceiver : public LooperCallback {
public:
NativeDisplayEventReceiver(JNIEnv* env,
jobject receiverObj, const sp<MessageQueue>& messageQueue);

status_t initialize();
void dispose();
status_t scheduleVsync();

protected:
Expand All @@ -59,7 +60,7 @@ class NativeDisplayEventReceiver : public RefBase {
DisplayEventReceiver mReceiver;
bool mWaitingForVsync;

static int handleReceiveCallback(int receiveFd, int events, void* data);
virtual int handleEvent(int receiveFd, int events, void* data);
bool readLastVsyncMessage(nsecs_t* outTimestamp, uint32_t* outCount);
};

Expand All @@ -72,12 +73,6 @@ NativeDisplayEventReceiver::NativeDisplayEventReceiver(JNIEnv* env,
}

NativeDisplayEventReceiver::~NativeDisplayEventReceiver() {
ALOGV("receiver %p ~ Disposing display event receiver.", this);

if (!mReceiver.initCheck()) {
mMessageQueue->getLooper()->removeFd(mReceiver.getFd());
}

JNIEnv* env = AndroidRuntime::getJNIEnv();
env->DeleteGlobalRef(mReceiverObjGlobal);
}
Expand All @@ -90,13 +85,21 @@ status_t NativeDisplayEventReceiver::initialize() {
}

int rc = mMessageQueue->getLooper()->addFd(mReceiver.getFd(), 0, ALOOPER_EVENT_INPUT,
handleReceiveCallback, this);
this, NULL);
if (rc < 0) {
return UNKNOWN_ERROR;
}
return OK;
}

void NativeDisplayEventReceiver::dispose() {
ALOGV("receiver %p ~ Disposing display event receiver.", this);

if (!mReceiver.initCheck()) {
mMessageQueue->getLooper()->removeFd(mReceiver.getFd());
}
}

status_t NativeDisplayEventReceiver::scheduleVsync() {
if (!mWaitingForVsync) {
ALOGV("receiver %p ~ Scheduling vsync.", this);
Expand All @@ -117,9 +120,7 @@ status_t NativeDisplayEventReceiver::scheduleVsync() {
return OK;
}

int NativeDisplayEventReceiver::handleReceiveCallback(int receiveFd, int events, void* data) {
sp<NativeDisplayEventReceiver> r = static_cast<NativeDisplayEventReceiver*>(data);

int NativeDisplayEventReceiver::handleEvent(int receiveFd, int events, void* data) {
if (events & (ALOOPER_EVENT_ERROR | ALOOPER_EVENT_HANGUP)) {
ALOGE("Display event receiver pipe was closed or an error occurred. "
"events=0x%x", events);
Expand All @@ -135,23 +136,23 @@ int NativeDisplayEventReceiver::handleReceiveCallback(int receiveFd, int events,
// Drain all pending events, keep the last vsync.
nsecs_t vsyncTimestamp;
uint32_t vsyncCount;
if (!r->readLastVsyncMessage(&vsyncTimestamp, &vsyncCount)) {
ALOGV("receiver %p ~ Woke up but there was no vsync pulse!", data);
if (!readLastVsyncMessage(&vsyncTimestamp, &vsyncCount)) {
ALOGV("receiver %p ~ Woke up but there was no vsync pulse!", this);
return 1; // keep the callback, did not obtain a vsync pulse
}

ALOGV("receiver %p ~ Vsync pulse: timestamp=%lld, count=%d",
data, vsyncTimestamp, vsyncCount);
r->mWaitingForVsync = false;
this, vsyncTimestamp, vsyncCount);
mWaitingForVsync = false;

JNIEnv* env = AndroidRuntime::getJNIEnv();

ALOGV("receiver %p ~ Invoking vsync handler.", data);
env->CallVoidMethod(r->mReceiverObjGlobal,
ALOGV("receiver %p ~ Invoking vsync handler.", this);
env->CallVoidMethod(mReceiverObjGlobal,
gDisplayEventReceiverClassInfo.dispatchVsync, vsyncTimestamp, vsyncCount);
ALOGV("receiver %p ~ Returned from vsync handler.", data);
ALOGV("receiver %p ~ Returned from vsync handler.", this);

r->mMessageQueue->raiseAndClearException(env, "dispatchVsync");
mMessageQueue->raiseAndClearException(env, "dispatchVsync");
return 1; // keep the callback
}

Expand Down Expand Up @@ -201,6 +202,7 @@ static jint nativeInit(JNIEnv* env, jclass clazz, jobject receiverObj,
static void nativeDispose(JNIEnv* env, jclass clazz, jint receiverPtr) {
sp<NativeDisplayEventReceiver> receiver =
reinterpret_cast<NativeDisplayEventReceiver*>(receiverPtr);
receiver->dispose();
receiver->decStrong(gDisplayEventReceiverClassInfo.clazz); // drop reference held by the object
}

Expand Down
35 changes: 18 additions & 17 deletions core/jni/android_view_InputEventReceiver.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -44,13 +44,14 @@ static struct {
} gInputEventReceiverClassInfo;


class NativeInputEventReceiver : public RefBase {
class NativeInputEventReceiver : public LooperCallback {
public:
NativeInputEventReceiver(JNIEnv* env,
jobject receiverObj, const sp<InputChannel>& inputChannel,
const sp<MessageQueue>& messageQueue);

status_t initialize();
void dispose();
status_t finishInputEvent(uint32_t seq, bool handled);
status_t consumeEvents(JNIEnv* env, bool consumeBatches, nsecs_t frameTime);

Expand All @@ -68,7 +69,7 @@ class NativeInputEventReceiver : public RefBase {
return mInputConsumer.getChannel()->getName().string();
}

static int handleReceiveCallback(int receiveFd, int events, void* data);
virtual int handleEvent(int receiveFd, int events, void* data);
};


Expand All @@ -84,23 +85,24 @@ NativeInputEventReceiver::NativeInputEventReceiver(JNIEnv* env,
}

NativeInputEventReceiver::~NativeInputEventReceiver() {
#if DEBUG_DISPATCH_CYCLE
ALOGD("channel '%s' ~ Disposing input event receiver.", getInputChannelName());
#endif

mMessageQueue->getLooper()->removeFd(mInputConsumer.getChannel()->getFd());

JNIEnv* env = AndroidRuntime::getJNIEnv();
env->DeleteGlobalRef(mReceiverObjGlobal);
}

status_t NativeInputEventReceiver::initialize() {
int receiveFd = mInputConsumer.getChannel()->getFd();
mMessageQueue->getLooper()->addFd(
receiveFd, 0, ALOOPER_EVENT_INPUT, handleReceiveCallback, this);
mMessageQueue->getLooper()->addFd(receiveFd, 0, ALOOPER_EVENT_INPUT, this, NULL);
return OK;
}

void NativeInputEventReceiver::dispose() {
#if DEBUG_DISPATCH_CYCLE
ALOGD("channel '%s' ~ Disposing input event receiver.", getInputChannelName());
#endif

mMessageQueue->getLooper()->removeFd(mInputConsumer.getChannel()->getFd());
}

status_t NativeInputEventReceiver::finishInputEvent(uint32_t seq, bool handled) {
#if DEBUG_DISPATCH_CYCLE
ALOGD("channel '%s' ~ Finished input event.", getInputChannelName());
Expand All @@ -114,24 +116,22 @@ status_t NativeInputEventReceiver::finishInputEvent(uint32_t seq, bool handled)
return status;
}

int NativeInputEventReceiver::handleReceiveCallback(int receiveFd, int events, void* data) {
sp<NativeInputEventReceiver> r = static_cast<NativeInputEventReceiver*>(data);

int NativeInputEventReceiver::handleEvent(int receiveFd, int events, void* data) {
if (events & (ALOOPER_EVENT_ERROR | ALOOPER_EVENT_HANGUP)) {
ALOGE("channel '%s' ~ Publisher closed input channel or an error occurred. "
"events=0x%x", r->getInputChannelName(), events);
"events=0x%x", getInputChannelName(), events);
return 0; // remove the callback
}

if (!(events & ALOOPER_EVENT_INPUT)) {
ALOGW("channel '%s' ~ Received spurious callback for unhandled poll event. "
"events=0x%x", r->getInputChannelName(), events);
"events=0x%x", getInputChannelName(), events);
return 1;
}

JNIEnv* env = AndroidRuntime::getJNIEnv();
status_t status = r->consumeEvents(env, false /*consumeBatches*/, -1);
r->mMessageQueue->raiseAndClearException(env, "handleReceiveCallback");
status_t status = consumeEvents(env, false /*consumeBatches*/, -1);
mMessageQueue->raiseAndClearException(env, "handleReceiveCallback");
return status == OK || status == NO_MEMORY ? 1 : 0;
}

Expand Down Expand Up @@ -256,6 +256,7 @@ static jint nativeInit(JNIEnv* env, jclass clazz, jobject receiverObj,
static void nativeDispose(JNIEnv* env, jclass clazz, jint receiverPtr) {
sp<NativeInputEventReceiver> receiver =
reinterpret_cast<NativeInputEventReceiver*>(receiverPtr);
receiver->dispose();
receiver->decStrong(gInputEventReceiverClassInfo.clazz); // drop reference held by the object
}

Expand Down

0 comments on commit 6e8e41a

Please sign in to comment.