blob: 0a56568243a5a6a87327e19875a3e1b52b2fc08c [file] [log] [blame]
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +00001/*
2 * This file is part of the flashrom project.
3 *
4 * Copyright (C) 2009 Urja Rannikko <urjaman@gmail.com>
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +00005 * Copyright (C) 2009,2010 Carl-Daniel Hailfinger
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +00006 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
20 */
21
22#include <stdio.h>
23#include <stdlib.h>
24#include <unistd.h>
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +000025#include <string.h>
26#include <ctype.h>
27#include <fcntl.h>
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +000028#include <sys/stat.h>
29#include <errno.h>
30#include <inttypes.h>
Patrick Georgie48654c2010-01-06 22:14:39 +000031#ifdef _WIN32
32#include <conio.h>
33#else
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +000034#include <termios.h>
Virgil-Adrian Teacada7c5452012-04-30 23:11:06 +000035#include <unistd.h>
36#include <sys/types.h>
37#include <sys/ioctl.h>
Patrick Georgie48654c2010-01-06 22:14:39 +000038#endif
Carl-Daniel Hailfinger5b997c32010-07-27 22:41:39 +000039#include "flash.h"
40#include "programmer.h"
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +000041
Patrick Georgie48654c2010-01-06 22:14:39 +000042fdtype sp_fd;
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +000043
44void __attribute__((noreturn)) sp_die(char *msg)
45{
46 perror(msg);
47 exit(1);
48}
49
Patrick Georgie48654c2010-01-06 22:14:39 +000050#ifndef _WIN32
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +000051struct baudentry {
52 int flag;
53 unsigned int baud;
54};
55
56/* I'd like if the C preprocessor could have directives in macros */
57#define BAUDENTRY(baud) { B##baud, baud },
58static const struct baudentry sp_baudtable[] = {
59 BAUDENTRY(9600)
60 BAUDENTRY(19200)
61 BAUDENTRY(38400)
62 BAUDENTRY(57600)
63 BAUDENTRY(115200)
64#ifdef B230400
65 BAUDENTRY(230400)
66#endif
67#ifdef B460800
68 BAUDENTRY(460800)
69#endif
70#ifdef B500000
71 BAUDENTRY(500000)
72#endif
73#ifdef B576000
74 BAUDENTRY(576000)
75#endif
76#ifdef B921600
77 BAUDENTRY(921600)
78#endif
79#ifdef B1000000
80 BAUDENTRY(1000000)
81#endif
82#ifdef B1152000
83 BAUDENTRY(1152000)
84#endif
85#ifdef B1500000
86 BAUDENTRY(1500000)
87#endif
88#ifdef B2000000
89 BAUDENTRY(2000000)
90#endif
91#ifdef B2500000
92 BAUDENTRY(2500000)
93#endif
94#ifdef B3000000
95 BAUDENTRY(3000000)
96#endif
97#ifdef B3500000
98 BAUDENTRY(3500000)
99#endif
100#ifdef B4000000
101 BAUDENTRY(4000000)
102#endif
103 {0, 0} /* Terminator */
104};
Patrick Georgie48654c2010-01-06 22:14:39 +0000105#endif
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000106
Patrick Georgie48654c2010-01-06 22:14:39 +0000107fdtype sp_openserport(char *dev, unsigned int baud)
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000108{
Patrick Georgie48654c2010-01-06 22:14:39 +0000109#ifdef _WIN32
110 HANDLE fd;
Uwe Hermann43959702010-03-13 17:28:29 +0000111 char *dev2 = dev;
Carl-Daniel Hailfinger9e3a6c42010-10-08 12:40:09 +0000112 if ((strlen(dev) > 3) && (tolower((unsigned char)dev[0]) == 'c') &&
113 (tolower((unsigned char)dev[1]) == 'o') &&
114 (tolower((unsigned char)dev[2]) == 'm')) {
Uwe Hermann43959702010-03-13 17:28:29 +0000115 dev2 = malloc(strlen(dev) + 5);
Stefan Tauner269de352011-07-12 22:35:21 +0000116 if (!dev2)
117 sp_die("Error: Out of memory");
Patrick Georgi06602c22010-01-26 20:58:40 +0000118 strcpy(dev2, "\\\\.\\");
Uwe Hermann43959702010-03-13 17:28:29 +0000119 strcpy(dev2 + 4, dev);
Patrick Georgi06602c22010-01-26 20:58:40 +0000120 }
Uwe Hermann43959702010-03-13 17:28:29 +0000121 fd = CreateFile(dev2, GENERIC_READ | GENERIC_WRITE, 0, NULL,
122 OPEN_EXISTING, 0, NULL);
Patrick Georgi06602c22010-01-26 20:58:40 +0000123 if (dev2 != dev)
124 free(dev2);
Patrick Georgie48654c2010-01-06 22:14:39 +0000125 if (fd == INVALID_HANDLE_VALUE) {
126 sp_die("Error: cannot open serial port");
127 }
128 DCB dcb;
129 if (!GetCommState(fd, &dcb)) {
130 sp_die("Error: Could not fetch serial port configuration");
131 }
132 switch (baud) {
133 case 9600: dcb.BaudRate = CBR_9600; break;
134 case 19200: dcb.BaudRate = CBR_19200; break;
135 case 38400: dcb.BaudRate = CBR_38400; break;
136 case 57600: dcb.BaudRate = CBR_57600; break;
137 case 115200: dcb.BaudRate = CBR_115200; break;
138 default: sp_die("Error: Could not set baud rate");
139 }
Uwe Hermann43959702010-03-13 17:28:29 +0000140 dcb.ByteSize = 8;
141 dcb.Parity = NOPARITY;
142 dcb.StopBits = ONESTOPBIT;
Patrick Georgie48654c2010-01-06 22:14:39 +0000143 if (!SetCommState(fd, &dcb)) {
144 sp_die("Error: Could not change serial port configuration");
145 }
146 return fd;
147#else
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000148 struct termios options;
149 int fd, i;
150 fd = open(dev, O_RDWR | O_NOCTTY | O_NDELAY);
151 if (fd < 0)
152 sp_die("Error: cannot open serial port");
153 fcntl(fd, F_SETFL, 0);
154 tcgetattr(fd, &options);
155 for (i = 0;; i++) {
156 if (sp_baudtable[i].baud == 0) {
157 close(fd);
Uwe Hermann43959702010-03-13 17:28:29 +0000158 msg_perr("Error: cannot configure for baudrate %d\n",
159 baud);
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000160 exit(1);
161 }
162 if (sp_baudtable[i].baud == baud) {
163 cfsetispeed(&options, sp_baudtable[i].flag);
164 cfsetospeed(&options, sp_baudtable[i].flag);
165 break;
166 }
167 }
168 options.c_cflag &= ~(PARENB | CSTOPB | CSIZE | CRTSCTS);
169 options.c_cflag |= (CS8 | CLOCAL | CREAD);
170 options.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG);
171 options.c_iflag &= ~(IXON | IXOFF | IXANY | ICRNL | IGNCR | INLCR);
172 options.c_oflag &= ~OPOST;
173 tcsetattr(fd, TCSANOW, &options);
174 return fd;
Patrick Georgie48654c2010-01-06 22:14:39 +0000175#endif
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000176}
177
Virgil-Adrian Teacada7c5452012-04-30 23:11:06 +0000178void sp_set_pin(enum SP_PIN pin, int val) {
179#ifdef _WIN32
180 DWORD ctl;
181
182 if(pin == PIN_TXD) {
183 ctl = val ? SETBREAK: CLRBREAK;
184 }
185 else if(pin == PIN_DTR) {
186 ctl = val ? SETDTR: CLRDTR;
187 }
188 else {
189 ctl = val ? SETRTS: CLRRTS;
190 }
191 EscapeCommFunction(sp_fd, ctl);
192#else
193 int ctl, s;
194
195 if(pin == PIN_TXD) {
196 ioctl(sp_fd, val ? TIOCSBRK : TIOCCBRK, 0);
197 }
198 else {
199 s = (pin == PIN_DTR) ? TIOCM_DTR : TIOCM_RTS;
200 ioctl(sp_fd, TIOCMGET, &ctl);
201
202 if (val) {
203 ctl |= s;
204 }
205 else {
206 ctl &= ~s;
207 }
208 ioctl(sp_fd, TIOCMSET, &ctl);
209 }
210#endif
211}
212
213int sp_get_pin(enum SP_PIN pin) {
214 int s;
215#ifdef _WIN32
216 DWORD ctl;
217
218 s = (pin == PIN_CTS) ? MS_CTS_ON : MS_DSR_ON;
219 GetCommModemStatus(sp_fd, &ctl);
220#else
221 int ctl;
222 s = (pin == PIN_CTS) ? TIOCM_CTS : TIOCM_DSR;
223 ioctl(sp_fd, TIOCMGET, &ctl);
224#endif
225
226 return ((ctl & s) ? 1 : 0);
227
228}
229
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000230void sp_flush_incoming(void)
231{
Patrick Georgie48654c2010-01-06 22:14:39 +0000232#ifdef _WIN32
233 PurgeComm(sp_fd, PURGE_RXCLEAR);
234#else
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000235 tcflush(sp_fd, TCIFLUSH);
Patrick Georgie48654c2010-01-06 22:14:39 +0000236#endif
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000237 return;
238}
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000239
David Hendricks8bb20212011-06-14 01:35:36 +0000240int serialport_shutdown(void *data)
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000241{
Patrick Georgie48654c2010-01-06 22:14:39 +0000242#ifdef _WIN32
243 CloseHandle(sp_fd);
244#else
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000245 close(sp_fd);
Patrick Georgie48654c2010-01-06 22:14:39 +0000246#endif
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000247 return 0;
248}
249
250int serialport_write(unsigned char *buf, unsigned int writecnt)
251{
Uwe Hermannd5e85d62011-07-03 19:44:12 +0000252#ifdef _WIN32
253 DWORD tmp = 0;
254#else
255 ssize_t tmp = 0;
256#endif
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000257
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000258 while (writecnt > 0) {
Patrick Georgie48654c2010-01-06 22:14:39 +0000259#ifdef _WIN32
260 WriteFile(sp_fd, buf, writecnt, &tmp, NULL);
261#else
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000262 tmp = write(sp_fd, buf, writecnt);
Patrick Georgie48654c2010-01-06 22:14:39 +0000263#endif
Carl-Daniel Hailfingerd2f007f2010-09-16 22:34:25 +0000264 if (tmp == -1) {
265 msg_perr("Serial port write error!\n");
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000266 return 1;
Carl-Daniel Hailfingerd2f007f2010-09-16 22:34:25 +0000267 }
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000268 if (!tmp)
Sean Nelsonebb4f5f2010-01-09 23:46:39 +0000269 msg_pdbg("Empty write\n");
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000270 writecnt -= tmp;
271 buf += tmp;
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000272 }
273
274 return 0;
275}
276
277int serialport_read(unsigned char *buf, unsigned int readcnt)
278{
Uwe Hermannd5e85d62011-07-03 19:44:12 +0000279#ifdef _WIN32
280 DWORD tmp = 0;
281#else
282 ssize_t tmp = 0;
283#endif
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000284
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000285 while (readcnt > 0) {
Patrick Georgie48654c2010-01-06 22:14:39 +0000286#ifdef _WIN32
287 ReadFile(sp_fd, buf, readcnt, &tmp, NULL);
288#else
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000289 tmp = read(sp_fd, buf, readcnt);
Patrick Georgie48654c2010-01-06 22:14:39 +0000290#endif
Carl-Daniel Hailfingerd2f007f2010-09-16 22:34:25 +0000291 if (tmp == -1) {
292 msg_perr("Serial port read error!\n");
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000293 return 1;
Carl-Daniel Hailfingerd2f007f2010-09-16 22:34:25 +0000294 }
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000295 if (!tmp)
Sean Nelsonebb4f5f2010-01-09 23:46:39 +0000296 msg_pdbg("Empty read\n");
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000297 readcnt -= tmp;
298 buf += tmp;
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000299 }
300
301 return 0;
302}