Switch port to nonblocking mode prior to closing

This commit is contained in:
Will Hedgecock 2021-12-01 13:50:46 -06:00
parent 9e0aab54f2
commit 901334d08a
24 changed files with 34 additions and 14 deletions

View File

@ -2,7 +2,7 @@
* SerialPort_Posix.c * SerialPort_Posix.c
* *
* Created on: Feb 25, 2012 * Created on: Feb 25, 2012
* Last Updated on: Nov 19, 2021 * Last Updated on: Dec 01, 2021
* Author: Will Hedgecock * Author: Will Hedgecock
* *
* Copyright (C) 2012-2021 Fazecast, Inc. * Copyright (C) 2012-2021 Fazecast, Inc.
@ -90,6 +90,10 @@ JNIEXPORT jobjectArray JNICALL Java_com_fazecast_jSerialComm_SerialPort_getCommP
driverBasedSearchForComPorts(&serialPorts, "/proc/tty/driver/mvebu_serial", "/dev/ttyMV"); driverBasedSearchForComPorts(&serialPorts, "/proc/tty/driver/mvebu_serial", "/dev/ttyMV");
lastDitchSearchForComPorts(&serialPorts); lastDitchSearchForComPorts(&serialPorts);
#elif defined(__FreeBSD__)
recursiveSearchForComPorts(&serialPorts, "/sys/devices/");
#elif defined(__sun__) || defined(__APPLE__) #elif defined(__sun__) || defined(__APPLE__)
searchForComPorts(&serialPorts); searchForComPorts(&serialPorts);
@ -457,12 +461,20 @@ JNIEXPORT jint JNICALL Java_com_fazecast_jSerialComm_SerialPort_waitForEvent(JNI
JNIEXPORT jlong JNICALL Java_com_fazecast_jSerialComm_SerialPort_closePortNative(JNIEnv *env, jobject obj, jlong serialPortPointer) JNIEXPORT jlong JNICALL Java_com_fazecast_jSerialComm_SerialPort_closePortNative(JNIEnv *env, jobject obj, jlong serialPortPointer)
{ {
// Unblock, unlock, and close the port // Force the port to enter non-blocking mode to ensure that any current reads return
struct termios options = { 0 };
serialPort *port = (serialPort*)(intptr_t)serialPortPointer; serialPort *port = (serialPort*)(intptr_t)serialPortPointer;
tcgetattr(port->handle, &options);
options.c_cc[VMIN] = 0;
options.c_cc[VTIME] = 0;
fcntl(port->handle, F_SETFL, O_NONBLOCK);
tcsetattr(port->handle, TCSANOW, &options);
tcsetattr(port->handle, TCSANOW, &options);
// Unblock, unlock, and close the port
fsync(port->handle); fsync(port->handle);
tcdrain(port->handle); tcdrain(port->handle);
tcflush(port->handle, TCIOFLUSH); tcflush(port->handle, TCIOFLUSH);
fcntl(port->handle, F_SETFL, O_NONBLOCK);
flock(port->handle, LOCK_UN | LOCK_NB); flock(port->handle, LOCK_UN | LOCK_NB);
while (close(port->handle) && (errno == EINTR)) while (close(port->handle) && (errno == EINTR))
errno = 0; errno = 0;

View File

@ -2,7 +2,7 @@
* SerialPort_Windows.c * SerialPort_Windows.c
* *
* Created on: Feb 25, 2012 * Created on: Feb 25, 2012
* Last Updated on: Nov 19, 2021 * Last Updated on: Dec 01, 2021
* Author: Will Hedgecock * Author: Will Hedgecock
* *
* Copyright (C) 2012-2021 Fazecast, Inc. * Copyright (C) 2012-2021 Fazecast, Inc.

View File

@ -232,16 +232,24 @@ long openPortNative(void)
long closePortNative(long serialPortFD) long closePortNative(long serialPortFD)
{ {
// Unblock, unlock, and close the port // Force the port to enter non-blocking mode to ensure that any current reads return
fsync(serialPortFD); struct termios options = { 0 };
tcdrain(serialPortFD); tcgetattr(serialPortFD, &options);
tcflush(serialPortFD, TCIOFLUSH); options.c_cc[VMIN] = 0;
fcntl(serialPortFD, F_SETFL, O_NONBLOCK); options.c_cc[VTIME] = 0;
flock(serialPortFD, LOCK_UN | LOCK_NB); fcntl(serialPortFD, F_SETFL, O_NONBLOCK);
while (close(serialPortFD) && (errno == EINTR)) tcsetattr(serialPortFD, TCSANOW, &options);
errno = 0; tcsetattr(serialPortFD, TCSANOW, &options);
serialPortFD = -1;
return -1; // Unblock, unlock, and close the port
fsync(serialPortFD);
tcdrain(serialPortFD);
tcflush(serialPortFD, TCIOFLUSH);
flock(serialPortFD, LOCK_UN | LOCK_NB);
while (close(serialPortFD) && (errno == EINTR))
errno = 0;
serialPortFD = -1;
return -1;
} }
int readBytes(long serialPortFD, char* buffer, long bytesToRead, long offset, int timeoutMode, int readTimeout) int readBytes(long serialPortFD, char* buffer, long bytesToRead, long offset, int timeoutMode, int readTimeout)