2011-10-05 15:38:16 +02:00
|
|
|
/*
|
|
|
|
* LPC 2000 Loader, http://www.pjrc.com/arm/lpc2k_pgm
|
|
|
|
* Copyright (c) 2004, PJRC.COM, LLC, <paul@pjrc.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 2
|
|
|
|
* of the License.
|
|
|
|
*
|
|
|
|
* 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* If this code fails to build, please provide at least the following
|
|
|
|
* information when requesting (free) technical support.
|
|
|
|
*
|
|
|
|
* 1: Complete copy of all messages during the build.
|
|
|
|
* 2: Output of "gtk-config --version"
|
|
|
|
* 3: Output of "gtk-config --libs"
|
|
|
|
* 4: Output of "gtk-config --cflags"
|
|
|
|
* 5: Output of "uname -a"
|
|
|
|
* 6: Version of GTK installed... eg, type: ls -l /lib/libgtk*
|
|
|
|
* 7: Other info... which linux distribution, version, other software
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <pwd.h>
|
|
|
|
#include <grp.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <termios.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#ifdef LINUX
|
|
|
|
#include <linux/serial.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "serial.h"
|
|
|
|
|
2013-06-21 04:00:45 +02:00
|
|
|
int port_fd = -1;
|
2011-10-05 15:38:16 +02:00
|
|
|
|
|
|
|
static tcflag_t baud_name_to_flags(const char *baud_name);
|
|
|
|
static void report_open_error(const char *filename, int err);
|
|
|
|
|
2013-06-21 04:00:45 +02:00
|
|
|
char *baud_rate = "115200";
|
2011-10-05 15:38:16 +02:00
|
|
|
|
|
|
|
int open_serial_port(const char *port_name)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
int r;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (port_fd >= 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
close(port_fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
port_fd = open(port_name, O_RDWR);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (port_fd < 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
report_open_error(port_name, errno);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
r = set_baud(baud_rate);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (r == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("Port \"%s\" opened at %s baud\r\n",
|
|
|
|
port_name, baud_rate);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("Port \"%s\" opened, unable to set baud to %s\r\n",
|
|
|
|
port_name, baud_rate);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef LINUX
|
|
|
|
{
|
|
|
|
struct serial_struct kernel_serial_settings;
|
|
|
|
/* attempt to set low latency mode, but don't worry if we can't */
|
|
|
|
r = ioctl(port_fd, TIOCGSERIAL, &kernel_serial_settings);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (r < 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
kernel_serial_settings.flags |= ASYNC_LOW_LATENCY;
|
|
|
|
ioctl(port_fd, TIOCSSERIAL, &kernel_serial_settings);
|
|
|
|
}
|
|
|
|
#endif
|
2011-10-05 15:38:16 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* if the port can't be opened, try to print as much info as
|
|
|
|
* possible, so the problem can be resolved (usually permissions)
|
|
|
|
*/
|
|
|
|
static void report_open_error(const char *filename, int err)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
struct stat info;
|
|
|
|
uid_t my_uid;
|
|
|
|
gid_t my_gid;
|
|
|
|
char my_uname[64], my_gname[64], file_uname[64], file_gname[64];
|
|
|
|
struct passwd *p;
|
|
|
|
struct group *g;
|
|
|
|
mode_t perm;
|
|
|
|
int r, perm_ok = 0;
|
|
|
|
|
|
|
|
printf("\r\n");
|
|
|
|
printf("Unable to open \"%s\"\r\n", filename);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (err == EACCES) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("You don't have permission to access %s\r\n", filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
r = stat(filename, &info);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (r < 0) {
|
|
|
|
if (errno == ENOENT) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("file %s does not exist\r\n", filename);
|
|
|
|
}
|
2013-06-24 22:37:35 +02:00
|
|
|
else if (errno == ELOOP) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("too many symbolic links\r\n");
|
|
|
|
}
|
2013-06-24 22:37:35 +02:00
|
|
|
else if (errno == EACCES) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("permission denied to get file status\r\n");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("Unable to get file status, err%d\r\n", errno);
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
my_uid = getuid();
|
|
|
|
my_gid = getgid();
|
|
|
|
|
|
|
|
p = getpwuid(my_uid);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (p) {
|
2013-06-21 04:00:45 +02:00
|
|
|
snprintf(my_uname, sizeof(my_uname),
|
|
|
|
"\"%s\" (gid=%d)", p->pw_name, (int)my_uid);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
snprintf(my_uname, sizeof(my_uname),
|
|
|
|
"(gid=%d)", (int)my_uid);
|
|
|
|
}
|
|
|
|
|
|
|
|
p = getpwuid(info.st_uid);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (p) {
|
2013-06-21 04:00:45 +02:00
|
|
|
snprintf(file_uname, sizeof(file_uname),
|
|
|
|
"\"%s\" (uid=%d)", p->pw_name, (int)info.st_uid);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
snprintf(file_uname, sizeof(file_uname),
|
|
|
|
"(uid=%d)", (int)info.st_uid);
|
|
|
|
}
|
|
|
|
|
|
|
|
g = getgrgid(my_gid);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (g) {
|
2013-06-21 04:00:45 +02:00
|
|
|
snprintf(my_gname, sizeof(my_gname),
|
|
|
|
"\"%s\" (gid=%d)", g->gr_name, (int)my_gid);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
snprintf(my_gname, sizeof(my_gname),
|
|
|
|
"(gid=%d)", (int)my_gid);
|
|
|
|
}
|
|
|
|
|
|
|
|
g = getgrgid(info.st_gid);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (g) {
|
2013-06-21 04:00:45 +02:00
|
|
|
snprintf(file_gname, sizeof(file_gname),
|
|
|
|
"\"%s\" (uid=%d)", g->gr_name, (int)info.st_gid);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
snprintf(file_gname, sizeof(file_gname),
|
|
|
|
"(uid=%d)", (int)info.st_gid);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* printf("%s is owned by: user %s, group %s\r\n",
|
|
|
|
filename, file_uname, file_gname); */
|
|
|
|
|
|
|
|
perm = info.st_mode;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((perm & S_IROTH) && (perm & S_IWOTH)) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("%s has read/write permission for everybody\r\n",
|
|
|
|
filename);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("%s is not read/write for everybody, so\r\n", filename);
|
|
|
|
printf(" you must match either user or group permission\r\n");
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((perm & S_IRUSR) && (perm & S_IWUSR)) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("%s has read/write permission for user %s\r\n",
|
|
|
|
filename, file_uname);
|
|
|
|
perm_ok = 1;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((perm & S_IRGRP) && (perm & S_IWGRP)) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("%s has read/write permission for group %s\r\n",
|
|
|
|
filename, file_gname);
|
|
|
|
perm_ok = 1;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (perm_ok == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("%s does not read/write permission for user or group!\r\n",
|
|
|
|
filename);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("Your access privs: user %s, group %s\r\n",
|
|
|
|
my_uname, my_gname);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("\r\n");
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int write_serial_port(const void *buf, int num)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
return(write(port_fd, buf, num));
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void input_flush_serial_port(void)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
tcflush(port_fd, TCIFLUSH);
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int read_serial_port_nb(unsigned char *buf, int bufsize)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
int num, flags;
|
2011-10-05 15:38:16 +02:00
|
|
|
|
2013-06-21 04:00:45 +02:00
|
|
|
flags = fcntl(port_fd, F_GETFL);
|
|
|
|
fcntl(port_fd, F_SETFL, flags | O_NONBLOCK);
|
|
|
|
num = read(port_fd, buf, bufsize);
|
|
|
|
fcntl(port_fd, F_SETFL, flags);
|
|
|
|
return num;
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int read_serial_port(unsigned char *buf, int bufsize)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
int num;
|
2011-10-05 15:38:16 +02:00
|
|
|
|
2013-06-21 04:00:45 +02:00
|
|
|
num = read(port_fd, buf, bufsize);
|
2011-10-05 15:38:16 +02:00
|
|
|
|
2013-06-21 04:00:45 +02:00
|
|
|
return num;
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void send_break_signal(void)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
tcsendbreak(port_fd, 0);
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void close_serial_port(void)
|
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
if (port_fd >= 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
close(port_fd);
|
|
|
|
port_fd = -1;
|
|
|
|
}
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
tcflag_t baud_name_to_flags(const char *baud_name)
|
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "230400") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B230400;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "115200") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B115200;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "57600") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B57600;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "38400") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B38400;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "19200") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B19200;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "9600") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B9600;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "4800") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B4800;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "2400") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B2400;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "1200") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B1200;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (strcmp(baud_name, "300") == 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return B300;
|
|
|
|
}
|
|
|
|
|
|
|
|
return B0;
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int set_baud(const char *baud_name)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
struct termios port_setting;
|
|
|
|
tcflag_t baud;
|
|
|
|
int r;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (port_fd < 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
baud = baud_name_to_flags(baud_name);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (baud == B0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
r = tcgetattr(port_fd, &port_setting);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (r != 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return -3;
|
|
|
|
}
|
|
|
|
|
|
|
|
port_setting.c_iflag = IGNBRK | IGNPAR;
|
|
|
|
port_setting.c_cflag = baud | CS8 | CREAD | HUPCL | CLOCAL;
|
|
|
|
port_setting.c_oflag = 0;
|
|
|
|
port_setting.c_lflag = 0;
|
|
|
|
r = tcsetattr(port_fd, TCSAFLUSH, &port_setting);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (r != 0) {
|
2013-06-21 04:00:45 +02:00
|
|
|
return -4;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Normally this should never be used... except to pass the port
|
|
|
|
// file descriptor to the GTK event monitoring loop. All other
|
|
|
|
// use of the serial port is supposed to happen in the file.
|
|
|
|
int serial_port_fd(void)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
return port_fd;
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void set_rts(int val)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
int flags;
|
|
|
|
int result;
|
|
|
|
|
|
|
|
result = ioctl(port_fd, TIOCMGET, &flags);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (result == -1) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("Error %i while reading port io flags\n", errno);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (val) {
|
2013-06-21 04:00:45 +02:00
|
|
|
flags |= TIOCM_RTS;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
flags &= ~(TIOCM_RTS);
|
|
|
|
}
|
|
|
|
|
|
|
|
result = ioctl(port_fd, TIOCMSET, &flags);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (result == -1) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("Error %i while setting port io flags\n", errno);
|
|
|
|
}
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void set_dtr(int val)
|
|
|
|
{
|
2013-06-21 04:00:45 +02:00
|
|
|
int flags;
|
|
|
|
int result;
|
|
|
|
|
|
|
|
result = ioctl(port_fd, TIOCMGET, &flags);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (result == -1) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("Error %i while reading port io flags\n", errno);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (val) {
|
2013-06-21 04:00:45 +02:00
|
|
|
flags |= TIOCM_DTR;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
flags &= ~(TIOCM_DTR);
|
|
|
|
}
|
|
|
|
|
|
|
|
result = ioctl(port_fd, TIOCMSET, &flags);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (result == -1) {
|
2013-06-21 04:00:45 +02:00
|
|
|
printf("Error %i while setting port io flags\n", errno);
|
|
|
|
}
|
2011-10-05 15:38:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|