blob: 18562beac5478cfe001d0d0729b7d8d152516036 [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
Stefan Taunerbf88be92013-04-01 00:45:08 +0000107/* Uses msg_perr to print the last system error.
108 * Prints "Error: " followed first by \c msg and then by the description of the last error retrieved via
109 * strerror() or FormatMessage() and ending with a linebreak. */
110static void msg_perr_strerror(const char *msg)
111{
112 msg_perr("Error: %s", msg);
113#ifdef _WIN32
114 char *lpMsgBuf;
115 DWORD nErr = GetLastError();
116 FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, nErr,
117 MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL);
118 msg_perr(lpMsgBuf);
119 /* At least some formatted messages contain a line break at the end. Make sure to always print one */
120 if (lpMsgBuf[strlen(lpMsgBuf)-1] != '\n')
121 msg_perr("\n");
122 LocalFree(lpMsgBuf);
123#else
124 msg_perr("%s\n", strerror(errno));
125#endif
126}
127
Patrick Georgie48654c2010-01-06 22:14:39 +0000128fdtype sp_openserport(char *dev, unsigned int baud)
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000129{
Patrick Georgie48654c2010-01-06 22:14:39 +0000130#ifdef _WIN32
131 HANDLE fd;
Uwe Hermann43959702010-03-13 17:28:29 +0000132 char *dev2 = dev;
Stefan Taunerbf88be92013-04-01 00:45:08 +0000133 if ((strlen(dev) > 3) &&
134 (tolower((unsigned char)dev[0]) == 'c') &&
Carl-Daniel Hailfinger9e3a6c42010-10-08 12:40:09 +0000135 (tolower((unsigned char)dev[1]) == 'o') &&
136 (tolower((unsigned char)dev[2]) == 'm')) {
Uwe Hermann43959702010-03-13 17:28:29 +0000137 dev2 = malloc(strlen(dev) + 5);
Niklas Söderlund2a95e872012-07-30 19:42:33 +0000138 if (!dev2) {
Stefan Taunerbf88be92013-04-01 00:45:08 +0000139 msg_perr_strerror("Out of memory: ");
Stefan Taunerbb4fed72012-09-01 21:47:19 +0000140 return SER_INV_FD;
Niklas Söderlund2a95e872012-07-30 19:42:33 +0000141 }
Patrick Georgi06602c22010-01-26 20:58:40 +0000142 strcpy(dev2, "\\\\.\\");
Uwe Hermann43959702010-03-13 17:28:29 +0000143 strcpy(dev2 + 4, dev);
Patrick Georgi06602c22010-01-26 20:58:40 +0000144 }
Uwe Hermann43959702010-03-13 17:28:29 +0000145 fd = CreateFile(dev2, GENERIC_READ | GENERIC_WRITE, 0, NULL,
146 OPEN_EXISTING, 0, NULL);
Patrick Georgi06602c22010-01-26 20:58:40 +0000147 if (dev2 != dev)
148 free(dev2);
Patrick Georgie48654c2010-01-06 22:14:39 +0000149 if (fd == INVALID_HANDLE_VALUE) {
Stefan Taunerbf88be92013-04-01 00:45:08 +0000150 msg_perr_strerror("Cannot open serial port: ");
Stefan Taunerbb4fed72012-09-01 21:47:19 +0000151 return SER_INV_FD;
Patrick Georgie48654c2010-01-06 22:14:39 +0000152 }
153 DCB dcb;
154 if (!GetCommState(fd, &dcb)) {
Stefan Taunerbf88be92013-04-01 00:45:08 +0000155 msg_perr_strerror("Could not fetch serial port configuration: ");
Stefan Taunerbb4fed72012-09-01 21:47:19 +0000156 return SER_INV_FD;
Patrick Georgie48654c2010-01-06 22:14:39 +0000157 }
158 switch (baud) {
159 case 9600: dcb.BaudRate = CBR_9600; break;
160 case 19200: dcb.BaudRate = CBR_19200; break;
161 case 38400: dcb.BaudRate = CBR_38400; break;
162 case 57600: dcb.BaudRate = CBR_57600; break;
163 case 115200: dcb.BaudRate = CBR_115200; break;
Niklas Söderlund2a95e872012-07-30 19:42:33 +0000164 default: msg_perr("Error: Could not set baud rate: %s\n", strerror(errno));
Stefan Taunerbb4fed72012-09-01 21:47:19 +0000165 return SER_INV_FD;
Patrick Georgie48654c2010-01-06 22:14:39 +0000166 }
Uwe Hermann43959702010-03-13 17:28:29 +0000167 dcb.ByteSize = 8;
168 dcb.Parity = NOPARITY;
169 dcb.StopBits = ONESTOPBIT;
Patrick Georgie48654c2010-01-06 22:14:39 +0000170 if (!SetCommState(fd, &dcb)) {
Stefan Taunerbf88be92013-04-01 00:45:08 +0000171 msg_perr_strerror("Could not change serial port configuration: ");
Stefan Taunerbb4fed72012-09-01 21:47:19 +0000172 return SER_INV_FD;
Patrick Georgie48654c2010-01-06 22:14:39 +0000173 }
174 return fd;
175#else
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000176 struct termios options;
177 int fd, i;
178 fd = open(dev, O_RDWR | O_NOCTTY | O_NDELAY);
Niklas Söderlund2a95e872012-07-30 19:42:33 +0000179 if (fd < 0) {
Stefan Taunerbf88be92013-04-01 00:45:08 +0000180 msg_perr_strerror("Cannot open serial port: ");
Stefan Taunerbb4fed72012-09-01 21:47:19 +0000181 return SER_INV_FD;
Niklas Söderlund2a95e872012-07-30 19:42:33 +0000182 }
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000183 fcntl(fd, F_SETFL, 0);
184 tcgetattr(fd, &options);
185 for (i = 0;; i++) {
186 if (sp_baudtable[i].baud == 0) {
187 close(fd);
Niklas Söderlund2a95e872012-07-30 19:42:33 +0000188 msg_perr("Error: cannot configure for baudrate %d\n", baud);
Stefan Taunerbb4fed72012-09-01 21:47:19 +0000189 return SER_INV_FD;
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000190 }
191 if (sp_baudtable[i].baud == baud) {
192 cfsetispeed(&options, sp_baudtable[i].flag);
193 cfsetospeed(&options, sp_baudtable[i].flag);
194 break;
195 }
196 }
197 options.c_cflag &= ~(PARENB | CSTOPB | CSIZE | CRTSCTS);
198 options.c_cflag |= (CS8 | CLOCAL | CREAD);
199 options.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG);
200 options.c_iflag &= ~(IXON | IXOFF | IXANY | ICRNL | IGNCR | INLCR);
201 options.c_oflag &= ~OPOST;
202 tcsetattr(fd, TCSANOW, &options);
203 return fd;
Patrick Georgie48654c2010-01-06 22:14:39 +0000204#endif
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000205}
206
Virgil-Adrian Teacada7c5452012-04-30 23:11:06 +0000207void sp_set_pin(enum SP_PIN pin, int val) {
208#ifdef _WIN32
209 DWORD ctl;
210
211 if(pin == PIN_TXD) {
212 ctl = val ? SETBREAK: CLRBREAK;
213 }
214 else if(pin == PIN_DTR) {
215 ctl = val ? SETDTR: CLRDTR;
216 }
217 else {
218 ctl = val ? SETRTS: CLRRTS;
219 }
220 EscapeCommFunction(sp_fd, ctl);
221#else
222 int ctl, s;
223
224 if(pin == PIN_TXD) {
225 ioctl(sp_fd, val ? TIOCSBRK : TIOCCBRK, 0);
226 }
227 else {
228 s = (pin == PIN_DTR) ? TIOCM_DTR : TIOCM_RTS;
229 ioctl(sp_fd, TIOCMGET, &ctl);
230
231 if (val) {
232 ctl |= s;
233 }
234 else {
235 ctl &= ~s;
236 }
237 ioctl(sp_fd, TIOCMSET, &ctl);
238 }
239#endif
240}
241
242int sp_get_pin(enum SP_PIN pin) {
243 int s;
244#ifdef _WIN32
245 DWORD ctl;
246
247 s = (pin == PIN_CTS) ? MS_CTS_ON : MS_DSR_ON;
248 GetCommModemStatus(sp_fd, &ctl);
249#else
250 int ctl;
251 s = (pin == PIN_CTS) ? TIOCM_CTS : TIOCM_DSR;
252 ioctl(sp_fd, TIOCMGET, &ctl);
253#endif
254
255 return ((ctl & s) ? 1 : 0);
256
257}
258
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000259void sp_flush_incoming(void)
260{
Patrick Georgie48654c2010-01-06 22:14:39 +0000261#ifdef _WIN32
262 PurgeComm(sp_fd, PURGE_RXCLEAR);
263#else
Niklas Söderlund7145a502012-09-07 07:07:07 +0000264 /* FIXME: error handling */
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000265 tcflush(sp_fd, TCIFLUSH);
Patrick Georgie48654c2010-01-06 22:14:39 +0000266#endif
Carl-Daniel Hailfingere51ea102009-11-23 19:20:11 +0000267 return;
268}
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000269
David Hendricks8bb20212011-06-14 01:35:36 +0000270int serialport_shutdown(void *data)
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000271{
Patrick Georgie48654c2010-01-06 22:14:39 +0000272#ifdef _WIN32
273 CloseHandle(sp_fd);
274#else
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000275 close(sp_fd);
Patrick Georgie48654c2010-01-06 22:14:39 +0000276#endif
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000277 return 0;
278}
279
280int serialport_write(unsigned char *buf, unsigned int writecnt)
281{
Uwe Hermannd5e85d62011-07-03 19:44:12 +0000282#ifdef _WIN32
283 DWORD tmp = 0;
284#else
285 ssize_t tmp = 0;
286#endif
Stefan Tauner62574aa2012-11-30 16:46:41 +0000287 unsigned int empty_writes = 250; /* results in a ca. 125ms timeout */
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000288
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000289 while (writecnt > 0) {
Patrick Georgie48654c2010-01-06 22:14:39 +0000290#ifdef _WIN32
291 WriteFile(sp_fd, buf, writecnt, &tmp, NULL);
292#else
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000293 tmp = write(sp_fd, buf, writecnt);
Patrick Georgie48654c2010-01-06 22:14:39 +0000294#endif
Carl-Daniel Hailfingerd2f007f2010-09-16 22:34:25 +0000295 if (tmp == -1) {
296 msg_perr("Serial port write error!\n");
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000297 return 1;
Carl-Daniel Hailfingerd2f007f2010-09-16 22:34:25 +0000298 }
Stefan Tauner62574aa2012-11-30 16:46:41 +0000299 if (!tmp) {
300 msg_pdbg2("Empty write\n");
301 empty_writes--;
302 programmer_delay(500);
303 if (empty_writes == 0) {
304 msg_perr("Serial port is unresponsive!\n");
305 return 1;
306 }
307 }
308 writecnt -= tmp;
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000309 buf += tmp;
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000310 }
311
312 return 0;
313}
314
315int serialport_read(unsigned char *buf, unsigned int readcnt)
316{
Uwe Hermannd5e85d62011-07-03 19:44:12 +0000317#ifdef _WIN32
318 DWORD tmp = 0;
319#else
320 ssize_t tmp = 0;
321#endif
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000322
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000323 while (readcnt > 0) {
Patrick Georgie48654c2010-01-06 22:14:39 +0000324#ifdef _WIN32
325 ReadFile(sp_fd, buf, readcnt, &tmp, NULL);
326#else
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000327 tmp = read(sp_fd, buf, readcnt);
Patrick Georgie48654c2010-01-06 22:14:39 +0000328#endif
Carl-Daniel Hailfingerd2f007f2010-09-16 22:34:25 +0000329 if (tmp == -1) {
330 msg_perr("Serial port read error!\n");
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000331 return 1;
Carl-Daniel Hailfingerd2f007f2010-09-16 22:34:25 +0000332 }
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000333 if (!tmp)
Sean Nelsonebb4f5f2010-01-09 23:46:39 +0000334 msg_pdbg("Empty read\n");
Patrick Georgi3b6237d2010-01-06 19:09:40 +0000335 readcnt -= tmp;
336 buf += tmp;
Carl-Daniel Hailfingerefa151e2010-01-06 16:09:10 +0000337 }
338
339 return 0;
340}