370 lines
7.3 KiB
C
370 lines
7.3 KiB
C
/*
|
|
* This file is part of the sigrok project.
|
|
*
|
|
* Copyright (C) 2010-2012 Bert Vermeulen <bert@biot.com>
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#ifdef _WIN32
|
|
#include <windows.h>
|
|
#else
|
|
#include <glob.h>
|
|
#include <termios.h>
|
|
#endif
|
|
#include <stdlib.h>
|
|
#include <glib.h>
|
|
#include "libsigrok.h"
|
|
#include "libsigrok-internal.h"
|
|
|
|
// FIXME: Must be moved, or rather passed as function argument.
|
|
#ifdef _WIN32
|
|
static HANDLE hdl;
|
|
#endif
|
|
|
|
static const char *serial_port_glob[] = {
|
|
/* Linux */
|
|
"/dev/ttyS*",
|
|
"/dev/ttyUSB*",
|
|
"/dev/ttyACM*",
|
|
/* MacOS X */
|
|
"/dev/ttys*",
|
|
"/dev/tty.USB-*",
|
|
"/dev/tty.Modem-*",
|
|
NULL,
|
|
};
|
|
|
|
SR_PRIV GSList *list_serial_ports(void)
|
|
{
|
|
GSList *ports;
|
|
|
|
#ifdef _WIN32
|
|
/* TODO */
|
|
ports = NULL;
|
|
ports = g_slist_append(ports, g_strdup("COM1"));
|
|
#else
|
|
glob_t g;
|
|
unsigned int i, j;
|
|
|
|
ports = NULL;
|
|
for (i = 0; serial_port_glob[i]; i++) {
|
|
if (glob(serial_port_glob[i], 0, NULL, &g))
|
|
continue;
|
|
for (j = 0; j < g.gl_pathc; j++)
|
|
ports = g_slist_append(ports, g_strdup(g.gl_pathv[j]));
|
|
globfree(&g);
|
|
}
|
|
#endif
|
|
|
|
return ports;
|
|
}
|
|
|
|
SR_PRIV int serial_open(const char *pathname, int flags)
|
|
{
|
|
#ifdef _WIN32
|
|
/* FIXME: Don't hardcode COM1. */
|
|
hdl = CreateFile("COM1", GENERIC_READ | GENERIC_WRITE, 0, 0,
|
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
|
|
if (hdl == INVALID_HANDLE_VALUE) {
|
|
/* TODO: Error handling. */
|
|
}
|
|
return 0;
|
|
#else
|
|
return open(pathname, flags);
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Close the serial port.
|
|
* Returns 0 upon success, -1 upon failure.
|
|
*/
|
|
SR_PRIV int serial_close(int fd)
|
|
{
|
|
#ifdef _WIN32
|
|
/* Returns non-zero upon success, 0 upon failure. */
|
|
return (CloseHandle(hdl) == 0) ? -1 : 0;
|
|
#else
|
|
/* Returns 0 upon success, -1 upon failure. */
|
|
return close(fd);
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Flush serial port buffers (if any).
|
|
* Returns 0 upon success, -1 upon failure.
|
|
*/
|
|
SR_PRIV int serial_flush(int fd)
|
|
{
|
|
#ifdef _WIN32
|
|
/* Returns non-zero upon success, 0 upon failure. */
|
|
return (PurgeComm(hdl, PURGE_RXCLEAR | PURGE_TXCLEAR) == 0) ? -1 : 0;
|
|
#else
|
|
/* Returns 0 upon success, -1 upon failure. */
|
|
return tcflush(fd, TCIOFLUSH);
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Write a number of bytes to the specified serial port.
|
|
* Returns the number of bytes written, or -1 upon failure.
|
|
*/
|
|
SR_PRIV int serial_write(int fd, const void *buf, size_t count)
|
|
{
|
|
#ifdef _WIN32
|
|
DWORD tmp = 0;
|
|
|
|
/* FIXME */
|
|
/* Returns non-zero upon success, 0 upon failure. */
|
|
WriteFile(hdl, buf, count, &tmp, NULL);
|
|
#else
|
|
/* Returns the number of bytes written, or -1 upon failure. */
|
|
return write(fd, buf, count);
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Read a number of bytes from the specified serial port.
|
|
* Returns the number of bytes read, or -1 upon failure.
|
|
*/
|
|
SR_PRIV int serial_read(int fd, void *buf, size_t count)
|
|
{
|
|
#ifdef _WIN32
|
|
DWORD tmp = 0;
|
|
|
|
/* FIXME */
|
|
/* Returns non-zero upon success, 0 upon failure. */
|
|
return ReadFile(hdl, buf, count, &tmp, NULL);
|
|
#else
|
|
/* Returns the number of bytes read, or -1 upon failure. */
|
|
return read(fd, buf, count);
|
|
#endif
|
|
}
|
|
|
|
SR_PRIV void *serial_backup_params(int fd)
|
|
{
|
|
#ifdef _WIN32
|
|
/* TODO */
|
|
#else
|
|
struct termios *term;
|
|
|
|
/* TODO: 'term' is never g_free()'d? */
|
|
if (!(term = g_try_malloc(sizeof(struct termios)))) {
|
|
sr_err("serial: %s: term malloc failed", __func__);
|
|
return NULL;
|
|
}
|
|
|
|
tcgetattr(fd, term);
|
|
|
|
return term;
|
|
#endif
|
|
}
|
|
|
|
SR_PRIV void serial_restore_params(int fd, void *backup)
|
|
{
|
|
#ifdef _WIN32
|
|
/* TODO */
|
|
#else
|
|
tcsetattr(fd, TCSADRAIN, (struct termios *)backup);
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Set serial parameters.
|
|
*
|
|
* flowcontrol: 1 = rts/cts, 2 = xon/xoff
|
|
* parity: 0 = none, 1 = even, 2 = odd
|
|
*/
|
|
SR_PRIV int serial_set_params(int fd, int baudrate, int bits, int parity,
|
|
int stopbits, int flowcontrol)
|
|
{
|
|
#ifdef _WIN32
|
|
DCB dcb;
|
|
|
|
if (!GetCommState(hdl, &dcb)) {
|
|
/* TODO: Error handling. */
|
|
return SR_ERR;
|
|
}
|
|
|
|
switch (baudrate) {
|
|
/* TODO: Support for higher baud rates. */
|
|
case 115200:
|
|
dcb.BaudRate = CBR_115200;
|
|
break;
|
|
case 57600:
|
|
dcb.BaudRate = CBR_57600;
|
|
break;
|
|
case 38400:
|
|
dcb.BaudRate = CBR_38400;
|
|
break;
|
|
case 19200:
|
|
dcb.BaudRate = CBR_19200;
|
|
break;
|
|
case 9600:
|
|
dcb.BaudRate = CBR_9600;
|
|
break;
|
|
default:
|
|
/* TODO: Error handling. */
|
|
break;
|
|
}
|
|
dcb.ByteSize = bits;
|
|
dcb.Parity = NOPARITY; /* TODO: Don't hardcode. */
|
|
dcb.StopBits = ONESTOPBIT; /* TODO: Don't hardcode. */
|
|
|
|
if (!SetCommState(hdl, &dcb)) {
|
|
/* TODO: Error handling. */
|
|
return SR_ERR;
|
|
}
|
|
#else
|
|
struct termios term;
|
|
speed_t baud;
|
|
|
|
if (tcgetattr(fd, &term) < 0)
|
|
return SR_ERR;
|
|
|
|
switch (baudrate) {
|
|
case 9600:
|
|
baud = B9600;
|
|
break;
|
|
case 38400:
|
|
baud = B38400;
|
|
break;
|
|
case 57600:
|
|
baud = B57600;
|
|
break;
|
|
case 115200:
|
|
baud = B115200;
|
|
break;
|
|
#ifndef __APPLE__
|
|
case 460800:
|
|
baud = B460800;
|
|
break;
|
|
#endif
|
|
default:
|
|
return SR_ERR;
|
|
}
|
|
if (cfsetospeed(&term, baud) < 0)
|
|
return SR_ERR;
|
|
if (cfsetispeed(&term, baud) < 0)
|
|
return SR_ERR;
|
|
|
|
term.c_cflag &= ~CSIZE;
|
|
switch (bits) {
|
|
case 8:
|
|
term.c_cflag |= CS8;
|
|
break;
|
|
case 7:
|
|
term.c_cflag |= CS7;
|
|
break;
|
|
default:
|
|
return SR_ERR;
|
|
}
|
|
|
|
term.c_cflag &= ~CSTOPB;
|
|
switch (stopbits) {
|
|
case 1:
|
|
break;
|
|
case 2:
|
|
term.c_cflag |= CSTOPB;
|
|
default:
|
|
return SR_ERR;
|
|
}
|
|
|
|
term.c_iflag &= ~(IXON | IXOFF);
|
|
term.c_cflag &= ~CRTSCTS;
|
|
switch (flowcontrol) {
|
|
case 0:
|
|
/* No flow control. */
|
|
break;
|
|
case 1:
|
|
term.c_cflag |= CRTSCTS;
|
|
case 2:
|
|
term.c_iflag |= IXON | IXOFF;
|
|
break;
|
|
default:
|
|
return SR_ERR;
|
|
}
|
|
|
|
term.c_iflag &= ~IGNPAR;
|
|
term.c_cflag &= ~(PARODD | PARENB);
|
|
switch (parity) {
|
|
case SERIAL_PARITY_NONE:
|
|
term.c_iflag |= IGNPAR;
|
|
break;
|
|
case SERIAL_PARITY_EVEN:
|
|
term.c_cflag |= PARENB;
|
|
break;
|
|
case SERIAL_PARITY_ODD:
|
|
term.c_cflag |= PARENB | PARODD;
|
|
break;
|
|
default:
|
|
return SR_ERR;
|
|
}
|
|
|
|
if (tcsetattr(fd, TCSADRAIN, &term) < 0)
|
|
return SR_ERR;
|
|
#endif
|
|
|
|
return SR_OK;
|
|
}
|
|
|
|
#define SERIAL_COMM_SPEC "^(\\d+)/([78])([neo])([12])$"
|
|
SR_PRIV int serial_set_paramstr(int fd, const char *paramstr)
|
|
{
|
|
GRegex *reg;
|
|
GMatchInfo *match;
|
|
int speed, databits, parity, stopbits;
|
|
char *mstr;
|
|
|
|
speed = databits = parity = stopbits = 0;
|
|
reg = g_regex_new(SERIAL_COMM_SPEC, 0, 0, NULL);
|
|
if (g_regex_match(reg, paramstr, 0, &match)) {
|
|
if ((mstr = g_match_info_fetch(match, 1)))
|
|
speed = strtoul(mstr, NULL, 10);
|
|
g_free(mstr);
|
|
if ((mstr = g_match_info_fetch(match, 2)))
|
|
databits = strtoul(mstr, NULL, 10);
|
|
g_free(mstr);
|
|
if ((mstr = g_match_info_fetch(match, 3))) {
|
|
switch (mstr[0]) {
|
|
case 'n':
|
|
parity = SERIAL_PARITY_NONE;
|
|
break;
|
|
case 'e':
|
|
parity = SERIAL_PARITY_EVEN;
|
|
break;
|
|
case 'o':
|
|
parity = SERIAL_PARITY_ODD;
|
|
break;
|
|
}
|
|
}
|
|
g_free(mstr);
|
|
if ((mstr = g_match_info_fetch(match, 4)))
|
|
stopbits = strtoul(mstr, NULL, 10);
|
|
g_free(mstr);
|
|
}
|
|
g_match_info_unref(match);
|
|
g_regex_unref(reg);
|
|
|
|
if (speed)
|
|
return serial_set_params(fd, speed, databits, parity, stopbits, 0);
|
|
else
|
|
return SR_ERR_ARG;
|
|
}
|
|
|