Próbuję odczytać i zapisać dane z płyty FPGA. Sama płyta została dostarczona wraz ze sterownikiem, który tworzy urządzenie terminalowe o nazwie ttyUSB0 za każdym razem, gdy płyta jest podłączona. W układzie FPGA zaimplementowano asynchroniczny odbiornik i nadajnik, i wydaje się, że działają.Jak prawidłowo skonfigurować komunikację szeregową w systemie Linux?
Jednak wydaje się, że problem dotyczy strony C rzeczy. Używam niektórych wektorów testowych do testowania, czy FPGA wyprowadza odpowiednie informacje. Zauważyłem kilka rzeczy rzeczy:
- Urządzenie czasami nie otwiera się prawidłowo
- Terminal atrybuty czasami nie być pobierane lub ustaw.
- Odczyt jest czasami nieblokujący i nie odzyskuje właściwej wartości.
Poniżej przedstawiono sposób konfigurowania opcji deskryptorów terminali i plików. Wiele z tego pochodzi stąd: http://slackware.osuosl.org/slackware-3.3/docs/mini/Serial-Port-Programming
Wszelkie rady lub komentarze na temat przyczyn niepowodzenia programu mogą być bardzo pomocne.
#include <stdio.h> // Standard input/output definitions
#include <string.h> // String function definitions
#include <unistd.h> // UNIX standard function definitions
#include <fcntl.h> // File control definitions
#include <errno.h> // Error number definitions
#include <termios.h> // POSIX terminal control definitions
int open_port(void){
int fd; // File descriptor for the port
fd = open("/dev/ttyUSB0", O_RDWR | O_NOCTTY);
if (fd == -1){
fprintf(stderr, "open_port: Unable to open /dev/ttyUSB0 %s\n",strerror(errno));
exit(EXIT_FAILURE);
}
return (fd);
}
int main(void){
int fd = 0; // File descriptor
struct termios options; // Terminal options
fd = open_port(); // Open tty device for RD and WR
fcntl(fd, F_SETFL); // Configure port reading
tcgetattr(fd, &options); // Get the current options for the port
cfsetispeed(&options, B230400); // Set the baud rates to 230400
cfsetospeed(&options, B230400);
options.c_cflag |= (CLOCAL | CREAD); // Enable the receiver and set local mode
options.c_cflag &= ~PARENB; // No parity bit
options.c_cflag &= ~CSTOPB; // 1 stop bit
options.c_cflag &= ~CSIZE; // Mask data size
options.c_cflag |= CS8; // Select 8 data bits
options.c_cflag &= ~CRTSCTS; // Disable hardware flow control
// Enable data to be processed as raw input
options.c_lflag &= ~(ICANON | ECHO | ISIG);
// Set the new attributes
tcsetattr(fd, TCSANOW, &options);
////////////////////////////////////
// Simple read and write code here//
////////////////////////////////////
// Close file descriptor & exit
close(fd)
return EXIT_SUCCESS
}
UPDATE I zostały zmodyfikowane mój kod na podstawie pierwszej odpowiedzi. To jest to, co mam teraz:
#include <errno.h> // Error number definitions
#include <stdint.h> // C99 fixed data types
#include <stdio.h> // Standard input/output definitions
#include <stdlib.h> // C standard library
#include <string.h> // String function definitions
#include <unistd.h> // UNIX standard function definitions
#include <fcntl.h> // File control definitions
#include <termios.h> // POSIX terminal control definitions
// Open usb-serial port for reading & writing
int open_port(void){
int fd; // File descriptor for the port
fd = open("/dev/ttyUSB0", O_RDWR | O_NOCTTY);
if (fd == -1){
fprintf(stderr, "open_port: Unable to open /dev/ttyUSB0 %s\n",strerror(errno));
exit(EXIT_FAILURE);
}
return fd;
}
int main(void){
int fd = 0; // File descriptor
struct termios options; // Terminal options
int rc; // Return value
fd = open_port(); // Open tty device for RD and WR
// Get the current options for the port
if((rc = tcgetattr(fd, &options)) < 0){
fprintf(stderr, "failed to get attr: %d, %s\n", fd, strerror(errno));
exit(EXIT_FAILURE);
}
// Set the baud rates to 230400
cfsetispeed(&options, B230400);
// Set the baud rates to 230400
cfsetospeed(&options, B230400);
cfmakeraw(&options);
options.c_cflag |= (CLOCAL | CREAD); // Enable the receiver and set local mode
options.c_cflag &= ~CSTOPB; // 1 stop bit
options.c_cflag &= ~CRTSCTS; // Disable hardware flow control
options.c_cc[VMIN] = 1;
options.c_cc[VTIME] = 2;
// Set the new attributes
if((rc = tcsetattr(fd, TCSANOW, &options)) < 0){
fprintf(stderr, "failed to set attr: %d, %s\n", fd, strerror(errno));
exit(EXIT_FAILURE);
}
////////////////////////////////
// Simple Read/Write Code Here//
////////////////////////////////
// Close file descriptor & exit
close(fd);
return EXIT_SUCCESS;
}
Właśnie w celu wyjaśnienia, odbiornik i nadajnik używanie 8 bitów danych, 1 bit stopu, bez bitu parzystości.
Dzięki, dam ci to i zaktualizuję. – sj755
Zrobiłem aktualizację i opublikowałem mój obecny kod. Z jakiegoś powodu program nie jest w stanie odczytać niczego z tablicy, po prostu czeka na funkcję odczytu. jakieś pomysły? – sj755
Nie potrzebujesz 'c_cflag & = ~ PARENB', ponieważ' cfmakeraw() 'obsłuży to. Prawdopodobnie potrzebujesz 'c_cflag & = ~ CSTOPB' i' c_cflag & = ~ CRTSCTS' który został usunięty! Każda z nich może zabić czytanie. – sawdust