source: trunk/serial.c @ 1525

Revision 1525, 6.0 KB checked in by mkarcher, 3 weeks ago (diff)

Add serial port bitbanging code

This adds the pony_spi driver which supports the SI_Prog adapter, which
is commonly used for SPI chips with PonyProg? 2000, and a custom adapter
called "SERBANG" which differs in the logic of two pins.

Signed-off-by: Virgil-Adrian Teaca <darkstarlinux@…>
Acked-by: Michael Karcher <flashrom@…>

Line 
1/*
2 * This file is part of the flashrom project.
3 *
4 * Copyright (C) 2009 Urja Rannikko <urjaman@gmail.com>
5 * Copyright (C) 2009,2010 Carl-Daniel Hailfinger
6 *
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>
25#include <string.h>
26#include <ctype.h>
27#include <fcntl.h>
28#include <sys/stat.h>
29#include <errno.h>
30#include <inttypes.h>
31#ifdef _WIN32
32#include <conio.h>
33#else
34#include <termios.h>
35#include <unistd.h>
36#include <sys/types.h>
37#include <sys/ioctl.h>
38#endif
39#include "flash.h"
40#include "programmer.h"
41
42fdtype sp_fd;
43
44void __attribute__((noreturn)) sp_die(char *msg)
45{
46        perror(msg);
47        exit(1);
48}
49
50#ifndef _WIN32
51struct 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};
105#endif
106
107fdtype sp_openserport(char *dev, unsigned int baud)
108{
109#ifdef _WIN32
110        HANDLE fd;
111        char *dev2 = dev;
112        if ((strlen(dev) > 3) && (tolower((unsigned char)dev[0]) == 'c') &&
113            (tolower((unsigned char)dev[1]) == 'o') &&
114            (tolower((unsigned char)dev[2]) == 'm')) {
115                dev2 = malloc(strlen(dev) + 5);
116                if (!dev2)
117                        sp_die("Error: Out of memory");
118                strcpy(dev2, "\\\\.\\");
119                strcpy(dev2 + 4, dev);
120        }
121        fd = CreateFile(dev2, GENERIC_READ | GENERIC_WRITE, 0, NULL,
122                        OPEN_EXISTING, 0, NULL);
123        if (dev2 != dev)
124                free(dev2);
125        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        }
140        dcb.ByteSize = 8;
141        dcb.Parity = NOPARITY;
142        dcb.StopBits = ONESTOPBIT;
143        if (!SetCommState(fd, &dcb)) {
144                sp_die("Error: Could not change serial port configuration");
145        }
146        return fd;
147#else
148        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);
158                        msg_perr("Error: cannot configure for baudrate %d\n",
159                                 baud);
160                        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;
175#endif
176}
177
178void 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
230void sp_flush_incoming(void)
231{
232#ifdef _WIN32
233        PurgeComm(sp_fd, PURGE_RXCLEAR);
234#else
235        tcflush(sp_fd, TCIFLUSH);
236#endif
237        return;
238}
239
240int serialport_shutdown(void *data)
241{
242#ifdef _WIN32
243        CloseHandle(sp_fd);
244#else
245        close(sp_fd);
246#endif
247        return 0;
248}
249
250int serialport_write(unsigned char *buf, unsigned int writecnt)
251{
252#ifdef _WIN32
253        DWORD tmp = 0;
254#else
255        ssize_t tmp = 0;
256#endif
257
258        while (writecnt > 0) {
259#ifdef _WIN32
260                WriteFile(sp_fd, buf, writecnt, &tmp, NULL);
261#else
262                tmp = write(sp_fd, buf, writecnt);
263#endif
264                if (tmp == -1) {
265                        msg_perr("Serial port write error!\n");
266                        return 1;
267                }
268                if (!tmp)
269                        msg_pdbg("Empty write\n");
270                writecnt -= tmp; 
271                buf += tmp;
272        }
273
274        return 0;
275}
276
277int serialport_read(unsigned char *buf, unsigned int readcnt)
278{
279#ifdef _WIN32
280        DWORD tmp = 0;
281#else
282        ssize_t tmp = 0;
283#endif
284
285        while (readcnt > 0) {
286#ifdef _WIN32
287                ReadFile(sp_fd, buf, readcnt, &tmp, NULL);
288#else
289                tmp = read(sp_fd, buf, readcnt);
290#endif
291                if (tmp == -1) {
292                        msg_perr("Serial port read error!\n");
293                        return 1;
294                }
295                if (!tmp)
296                        msg_pdbg("Empty read\n");
297                readcnt -= tmp;
298                buf += tmp;
299        }
300
301        return 0;
302}
Note: See TracBrowser for help on using the repository browser.