Serial Comm usando WriteFile / ReadFile

//#include "StdAfx.h" #include  #include  #include  #include  #include  using namespace std; int main() { int com = 'COM2'; string data = "\n 010400 \n"; char output[32]; //unsigned int length = 0; DCB config = {0}; bool abContinue = true; DWORD dwBytesWritten; DWORD dwBytesRead; int isRead = false; HANDLE m_hCommPort = ::CreateFile(L"COM2", GENERIC_READ|GENERIC_WRITE,//access ( read and write) 0, //(share) 0:cannot share the COM port 0, //security (None) OPEN_EXISTING,// creation : open_existing 0, // we dont want overlapped operation 0// no templates file for COM port... ); config.DCBlength = sizeof(config); if((GetCommState(m_hCommPort, &config) == 0)) { printf("Get configuration port has a problem."); return FALSE; } config.BaudRate = 9600; config.StopBits = ONESTOPBIT; config.Parity = PARITY_NONE; config.ByteSize = DATABITS_8; config.fDtrControl = 0; config.fRtsControl = 0; if (!SetCommState(m_hCommPort, &config)) { printf( "Failed to Set Comm State Reason: %d\n",GetLastError()); //return E_FAIL; } printf("Current Settings\n Baud Rate %d\n Parity %d\n Byte Size %d\n Stop Bits %d", config.BaudRate, config.Parity, config.ByteSize, config.StopBits); int isWritten = WriteFile(m_hCommPort, &data,(DWORD) sizeof(data), &dwBytesWritten, NULL); //memset(output, 0, sizeof(output)); while (abContinue) { isRead = ReadFile(m_hCommPort, output, sizeof(output), &dwBytesRead, NULL); if(!isRead) { abContinue = false; break; } } cin.get(); } 

Tengo problemas para leer desde el puerto de comunicación. Si paso por el código, entra en “isRead = ReadFile (m_hCommPort, output, sizeof (output), & dwBytesRead, NULL);” y no vuelve … Este es mi primer bash sin éxito.

Puede intentar con algún código algo así después de abrir el archivo, pero antes de intentar usarlo:

 COMMTIMEOUTS timeouts; timeouts.ReadIntervalTimeout = 1; timeouts.ReadTotalTimeoutMultiplier = 1; timeouts.ReadTotalTimeoutConstant = 1; timeouts.WriteTotalTimeoutMultiplier = 1; timeouts.WriteTotalTimeoutConstant = 1; if (!SetCommTimeouts(m_hCommPort, &timeouts)) // setting timeouts failed. 

Editar: tal vez sea más fácil comenzar con un código que funcione y hacer que haga lo que quiera en lugar de intentar que el código funcione. Aquí hay un progtwig de terminal simple. Es minimalista en extremo, pero funciona (al menos lo suficientemente bien como para dejarme ver la salida de mi GPS, por ejemplo). Está muy lejos de lo que cualquiera (y mucho menos yo) llamaría sofisticado, pero debería dar al menos alguna idea de cómo empezar.

 #include  #include  #include  #define STRICT #define WIN32_LEAN_AND_MEAN #include  void system_error(char *name) { // Retrieve, format, and print out a message from the last error. The // `name' that's passed should be in the form of a present tense noun // (phrase) such as "opening file". // char *ptr = NULL; FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, 0, GetLastError(), 0, (char *)&ptr, 1024, NULL); fprintf(stderr, "\nError %s: %s\n", name, ptr); LocalFree(ptr); } int main(int argc, char **argv) { int ch; char buffer[1]; HANDLE file; COMMTIMEOUTS timeouts; DWORD read, written; DCB port; HANDLE keyboard = GetStdHandle(STD_INPUT_HANDLE); HANDLE screen = GetStdHandle(STD_OUTPUT_HANDLE); DWORD mode; char port_name[128] = "\\\\.\\COM3"; char init[] = ""; // eg, "ATZ" to completely reset a modem. if ( argc > 2 ) sprintf(port_name, "\\\\.\\COM%c", argv[1][0]); // open the comm port. file = CreateFile(port_name, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, NULL); if ( INVALID_HANDLE_VALUE == file) { system_error("opening file"); return 1; } // get the current DCB, and adjust a few bits to our liking. memset(&port, 0, sizeof(port)); port.DCBlength = sizeof(port); if ( !GetCommState(file, &port)) system_error("getting comm state"); if (!BuildCommDCB("baud=19200 parity=n data=8 stop=1", &port)) system_error("building comm DCB"); if (!SetCommState(file, &port)) system_error("adjusting port settings"); // set short timeouts on the comm port. timeouts.ReadIntervalTimeout = 1; timeouts.ReadTotalTimeoutMultiplier = 1; timeouts.ReadTotalTimeoutConstant = 1; timeouts.WriteTotalTimeoutMultiplier = 1; timeouts.WriteTotalTimeoutConstant = 1; if (!SetCommTimeouts(file, &timeouts)) system_error("setting port time-outs."); // set keyboard to raw reading. if (!GetConsoleMode(keyboard, &mode)) system_error("getting keyboard mode"); mode &= ~ ENABLE_PROCESSED_INPUT; if (!SetConsoleMode(keyboard, mode)) system_error("setting keyboard mode"); if (!EscapeCommFunction(file, CLRDTR)) system_error("clearing DTR"); Sleep(200); if (!EscapeCommFunction(file, SETDTR)) system_error("setting DTR"); if ( !WriteFile(file, init, sizeof(init), &written, NULL)) system_error("writing data to port"); if (written != sizeof(init)) system_error("not all data written to port"); // basic terminal loop: do { // check for data on port and display it on screen. ReadFile(file, buffer, sizeof(buffer), &read, NULL); if ( read ) WriteFile(screen, buffer, read, &written, NULL); // check for keypress, and write any out the port. if ( kbhit() ) { ch = getch(); WriteFile(file, &ch, 1, &written, NULL); } // until user hits ctrl-backspace. } while ( ch != 127); // close up and go home. CloseHandle(keyboard); CloseHandle(file); return 0; } 

La función ReadFile puede estar bloqueando el hilo, de ser así, permanecerá bloqueado hasta que se puedan leer algunos datos del puerto serie. Aquí hay un enlace para ver si es útil. Buena suerte.

Si no establece explícitamente los tiempos de espera , entonces ReadFile bloqueará indefinidamente hasta que los datos estén disponibles.

Tuve este problema en un archivo de lectura, con los tiempos de espera establecidos. Esto me estaba sacando crackers, así que terminé obteniendo un código de la web que funcionaba y luego cambiaba línea por línea para ver dónde estaba el error.

Resulta que el archivo de lectura estaba bien. Mi problema era un evento WaitCommEvent que se colgó cuando se desconectó el puerto ya que nunca se recibió ningún evento com …