// 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 #include #include #include "HALUtil.h" #include "edu_wpi_first_hal_can_CANJNI.h" #include "hal/CAN.h" using namespace hal; using namespace wpi::java; extern "C" { /* * Class: edu_wpi_first_hal_can_CANJNI * Method: FRCNetCommCANSessionMuxSendMessage * Signature: (I[BI)V */ JNIEXPORT void JNICALL Java_edu_wpi_first_hal_can_CANJNI_FRCNetCommCANSessionMuxSendMessage (JNIEnv* env, jclass, jint messageID, jbyteArray data, jint periodMs) { JSpan dataArray{env, data}; const uint8_t* dataBuffer = reinterpret_cast(dataArray.data()); uint8_t dataSize = dataArray.size(); int32_t status = 0; HAL_CAN_SendMessage(messageID, dataBuffer, dataSize, periodMs, &status); CheckCANStatus(env, status, messageID); } /* * Class: edu_wpi_first_hal_can_CANJNI * Method: FRCNetCommCANSessionMuxReceiveMessage * Signature: (Ljava/lang/Object;ILjava/lang/Object;)[B */ JNIEXPORT jbyteArray JNICALL Java_edu_wpi_first_hal_can_CANJNI_FRCNetCommCANSessionMuxReceiveMessage (JNIEnv* env, jclass, jobject messageID, jint messageIDMask, jobject timeStamp) { uint32_t* messageIDPtr = reinterpret_cast(env->GetDirectBufferAddress(messageID)); uint32_t* timeStampPtr = reinterpret_cast(env->GetDirectBufferAddress(timeStamp)); uint8_t dataSize = 0; uint8_t buffer[8]; int32_t status = 0; HAL_CAN_ReceiveMessage(messageIDPtr, messageIDMask, buffer, &dataSize, timeStampPtr, &status); if (!CheckCANStatus(env, status, *messageIDPtr)) { return nullptr; } return MakeJByteArray(env, {buffer, static_cast(dataSize)}); } /* * Class: edu_wpi_first_hal_can_CANJNI * Method: getCANStatus * Signature: (Ljava/lang/Object;)V */ JNIEXPORT void JNICALL Java_edu_wpi_first_hal_can_CANJNI_getCANStatus (JNIEnv* env, jclass, jobject canStatus) { float percentBusUtilization = 0; uint32_t busOffCount = 0; uint32_t txFullCount = 0; uint32_t receiveErrorCount = 0; uint32_t transmitErrorCount = 0; int32_t status = 0; HAL_CAN_GetCANStatus(&percentBusUtilization, &busOffCount, &txFullCount, &receiveErrorCount, &transmitErrorCount, &status); if (!CheckStatus(env, status)) { return; } SetCanStatusObject(env, canStatus, percentBusUtilization, busOffCount, txFullCount, receiveErrorCount, transmitErrorCount); } /* * Class: edu_wpi_first_hal_can_CANJNI * Method: openCANStreamSession * Signature: (III)I */ JNIEXPORT jint JNICALL Java_edu_wpi_first_hal_can_CANJNI_openCANStreamSession (JNIEnv* env, jclass, jint messageID, jint messageIDMask, jint maxMessages) { uint32_t handle = 0; int32_t status = 0; HAL_CAN_OpenStreamSession(&handle, static_cast(messageID), static_cast(messageIDMask), static_cast(maxMessages), &status); if (!CheckStatus(env, status)) { return static_cast(0); } return static_cast(handle); } /* * Class: edu_wpi_first_hal_can_CANJNI * Method: closeCANStreamSession * Signature: (I)V */ JNIEXPORT void JNICALL Java_edu_wpi_first_hal_can_CANJNI_closeCANStreamSession (JNIEnv* env, jclass, jint sessionHandle) { HAL_CAN_CloseStreamSession(static_cast(sessionHandle)); } /* * Class: edu_wpi_first_hal_can_CANJNI * Method: readCANStreamSession * Signature: (I[Ljava/lang/Object;I)I */ JNIEXPORT jint JNICALL Java_edu_wpi_first_hal_can_CANJNI_readCANStreamSession (JNIEnv* env, jclass, jint sessionHandle, jobjectArray messages, jint messagesToRead) { if (messages == nullptr) { ThrowNullPointerException(env, "messages cannot be null"); return 0; } jsize messagesArrayLen = env->GetArrayLength(messages); if (messagesArrayLen < messagesToRead) { messagesToRead = messagesArrayLen; } uint32_t handle = static_cast(sessionHandle); uint32_t messagesRead = 0; wpi::SmallVector messageBuffer; messageBuffer.resize_for_overwrite(messagesToRead); int32_t status = 0; HAL_CAN_ReadStreamSession(handle, messageBuffer.begin(), static_cast(messagesToRead), &messagesRead, &status); if (status == HAL_ERR_CANSessionMux_MessageNotFound || messagesRead == 0) { return 0; } if (status != HAL_ERR_CANSessionMux_SessionOverrun && !CheckStatus(env, status)) { return 0; } for (int i = 0; i < static_cast(messagesRead); i++) { struct HAL_CANStreamMessage* msg = &messageBuffer[i]; JLocal elem{ env, static_cast(env->GetObjectArrayElement(messages, i))}; if (!elem) { // If element doesn't exist, construct it in place. If that fails, we are // OOM, just return elem = JLocal{env, CreateCANStreamMessage(env)}; if (elem) { env->SetObjectArrayElement(messages, i, elem); } else { return 0; } } JLocal toSetArray{ env, SetCANStreamObject(env, elem, msg->dataSize, msg->messageID, msg->timeStamp)}; auto javaLen = env->GetArrayLength(toSetArray); if (javaLen < msg->dataSize) { msg->dataSize = javaLen; } env->SetByteArrayRegion(toSetArray, 0, msg->dataSize, reinterpret_cast(msg->data)); } if (status == HAL_ERR_CANSessionMux_SessionOverrun) { ThrowCANStreamOverflowException(env, messages, static_cast(messagesRead)); return 0; } return static_cast(messagesRead); } } // extern "C"