// Copyright (c) FIRST and other WPILib contributors. // Open Source Software; you can modify and/or share it under the terms of // the WPILib BSD license file in the root directory of this project. #include "BufferCallbackStore.h" #include #include #include "SimulatorJNI.h" #include "hal/Types.h" #include "hal/Value.h" #include "hal/handles/UnlimitedHandleResource.h" #include "hal/simulation/NotifyListener.h" using namespace hal; using namespace hal::sim; using namespace wpi::java; static hal::UnlimitedHandleResource* callbackHandles; namespace hal::sim { void InitializeBufferStore() { static hal::UnlimitedHandleResource cb; callbackHandles = &cb; } } // namespace hal::sim void BufferCallbackStore::create(JNIEnv* env, jobject obj) { m_call = JGlobal(env, obj); } void BufferCallbackStore::performCallback(const char* name, uint8_t* buffer, uint32_t length) { JNIEnv* env; JavaVM* vm = sim::GetJVM(); bool didAttachThread = false; int tryGetEnv = vm->GetEnv(reinterpret_cast(&env), JNI_VERSION_1_6); if (tryGetEnv == JNI_EDETACHED) { // Thread not attached didAttachThread = true; if (vm->AttachCurrentThread(reinterpret_cast(&env), nullptr) != 0) { // Failed to attach, log and return wpi::outs() << "Failed to attach\n"; wpi::outs().flush(); return; } } else if (tryGetEnv == JNI_EVERSION) { wpi::outs() << "Invalid JVM Version requested\n"; wpi::outs().flush(); } auto toCallbackArr = MakeJByteArray(env, wpi::StringRef{reinterpret_cast(buffer), static_cast(length)}); env->CallVoidMethod(m_call, sim::GetBufferCallback(), MakeJString(env, name), toCallbackArr, static_cast(length)); jbyte* fromCallbackArr = reinterpret_cast( env->GetPrimitiveArrayCritical(toCallbackArr, nullptr)); for (size_t i = 0; i < length; i++) { buffer[i] = fromCallbackArr[i]; } env->ReleasePrimitiveArrayCritical(toCallbackArr, fromCallbackArr, JNI_ABORT); if (env->ExceptionCheck()) { env->ExceptionDescribe(); } if (didAttachThread) { vm->DetachCurrentThread(); } } void BufferCallbackStore::free(JNIEnv* env) { m_call.free(env); } SIM_JniHandle sim::AllocateBufferCallback( JNIEnv* env, jint index, jobject callback, RegisterBufferCallbackFunc createCallback) { auto callbackStore = std::make_shared(); auto handle = callbackHandles->Allocate(callbackStore); if (handle == HAL_kInvalidHandle) { return -1; } uintptr_t handleAsPtr = static_cast(handle); void* handleAsVoidPtr = reinterpret_cast(handleAsPtr); callbackStore->create(env, callback); auto callbackFunc = [](const char* name, void* param, uint8_t* buffer, uint32_t length) { uintptr_t handleTmp = reinterpret_cast(param); SIM_JniHandle handle = static_cast(handleTmp); auto data = callbackHandles->Get(handle); if (!data) { return; } data->performCallback(name, buffer, length); }; auto id = createCallback(index, callbackFunc, handleAsVoidPtr); callbackStore->setCallbackId(id); return handle; } void sim::FreeBufferCallback(JNIEnv* env, SIM_JniHandle handle, jint index, FreeBufferCallbackFunc freeCallback) { auto callback = callbackHandles->Free(handle); freeCallback(index, callback->getCallbackId()); callback->free(env); }