// 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 "HALUtil.hpp" #include "org_wpilib_hardware_hal_can_CANJNI.h" #include "wpi/hal/CAN.h" #include "wpi/hal/Errors.h" #include "wpi/util/jni_util.hpp" using namespace wpi::hal; using namespace wpi::util::java; extern "C" { static bool PackCANMessage(JNIEnv* env, jbyteArray data, jint dataLength, jint flags, HAL_CANMessage* message) { if (data == nullptr) { ThrowNullPointerException(env, "data array cannot be null"); return false; } auto arrLen = env->GetArrayLength(data); if (arrLen < dataLength) { ThrowIllegalArgumentException(env, "array length less than data length"); return false; } if ((flags & HAL_CAN_FD_DATALENGTH) && dataLength > 64) { ThrowIllegalArgumentException(env, "FD frame has max length of 64 bytes"); return false; } else if (!(flags & HAL_CAN_FD_DATALENGTH) && dataLength > 8) { ThrowIllegalArgumentException(env, "Non FD frame has max length of 8 bytes"); return false; } std::memset(message, 0, sizeof(*message)); message->dataSize = dataLength; message->flags = flags; JSpan arr{env, data, static_cast(dataLength)}; std::memcpy(message->data, arr.data(), dataLength); return true; } /* * Class: org_wpilib_hardware_hal_can_CANJNI * Method: sendMessage * Signature: (II[BIII)I */ JNIEXPORT jint JNICALL Java_org_wpilib_hardware_hal_can_CANJNI_sendMessage (JNIEnv* env, jclass, jint busId, jint messageId, jbyteArray data, jint dataLength, jint flags, jint periodMs) { HAL_CANMessage message; if (!PackCANMessage(env, data, dataLength, flags, &message)) { return HAL_PARAMETER_OUT_OF_RANGE; } int32_t status = 0; HAL_CAN_SendMessage(busId, messageId, &message, periodMs, &status); return status; } /* * Class: org_wpilib_hardware_hal_can_CANJNI * Method: receiveMessage * Signature: (IILjava/lang/Object;)I */ JNIEXPORT jint JNICALL Java_org_wpilib_hardware_hal_can_CANJNI_receiveMessage (JNIEnv* env, jclass, jint busId, jint messageId, jobject data) { HAL_CANReceiveMessage message; std::memset(&message, 0, sizeof(message)); int32_t status = 0; HAL_CAN_ReceiveMessage(busId, messageId, &message, &status); jbyteArray toSetArray = SetCANReceiveMessageObject(env, data, message.message.dataSize, message.message.flags, message.timeStamp); auto javaLen = env->GetArrayLength(toSetArray); if (javaLen < message.message.dataSize) { ThrowIllegalArgumentException(env, "Message buffer not long enough for message"); return status; } env->SetByteArrayRegion(toSetArray, 0, message.message.dataSize, reinterpret_cast(message.message.data)); return status; } /* * Class: org_wpilib_hardware_hal_can_CANJNI * Method: getCANStatus * Signature: (ILjava/lang/Object;)V */ JNIEXPORT void JNICALL Java_org_wpilib_hardware_hal_can_CANJNI_getCANStatus (JNIEnv* env, jclass, jint busId, 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(busId, &percentBusUtilization, &busOffCount, &txFullCount, &receiveErrorCount, &transmitErrorCount, &status); if (!CheckStatus(env, status)) { return; } SetCanStatusObject(env, canStatus, percentBusUtilization, busOffCount, txFullCount, receiveErrorCount, transmitErrorCount); } /* * Class: org_wpilib_hardware_hal_can_CANJNI * Method: openCANStreamSession * Signature: (IIII)I */ JNIEXPORT jint JNICALL Java_org_wpilib_hardware_hal_can_CANJNI_openCANStreamSession (JNIEnv* env, jclass, jint busId, jint messageId, jint messageIDMask, jint maxMessages) { int32_t status = 0; HAL_CANStreamHandle handle = HAL_CAN_OpenStreamSession(busId, static_cast(messageId), static_cast(messageIDMask), static_cast(maxMessages), &status); if (!CheckStatus(env, status)) { return static_cast(0); } return static_cast(handle); } /* * Class: org_wpilib_hardware_hal_can_CANJNI * Method: closeCANStreamSession * Signature: (I)V */ JNIEXPORT void JNICALL Java_org_wpilib_hardware_hal_can_CANJNI_closeCANStreamSession (JNIEnv* env, jclass, jint sessionHandle) { HAL_CAN_CloseStreamSession(static_cast(sessionHandle)); } /* * Class: org_wpilib_hardware_hal_can_CANJNI * Method: readCANStreamSession * Signature: (I[Ljava/lang/Object;I)I */ JNIEXPORT jint JNICALL Java_org_wpilib_hardware_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::util::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->message.message.flags, msg->message.message.dataSize, msg->messageId, msg->message.timeStamp)}; auto javaLen = env->GetArrayLength(toSetArray); if (javaLen < msg->message.message.dataSize) { ThrowIllegalArgumentException( env, "Message buffer not long enough for message"); return HAL_PARAMETER_OUT_OF_RANGE; } env->SetByteArrayRegion( toSetArray, 0, msg->message.message.dataSize, reinterpret_cast(msg->message.message.data)); } if (status == HAL_ERR_CANSessionMux_SessionOverrun) { ThrowCANStreamOverflowException(env, messages, static_cast(messagesRead)); return 0; } return static_cast(messagesRead); } } // extern "C"