2 * Proxmark3 generic uart / rs232/ serial port library
4 * Copyright (c) 2012, Roel Verdult
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions are met:
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * 3. Neither the name of the copyright holders nor the
15 * names of its contributors may be used to endorse or promote products
16 * derived from this software without specific prior written permission.
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY
19 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
22 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
25 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 * Partly based on uart-code written by Teunis van Beelen, available:
33 * http://www.teuniz.net/RS-232/index.html
40 // Test if we are dealing with unix operating systems
44 typedef struct termios term_info
;
46 int fd
; // Serial port file descriptor
47 term_info tiOld
; // Terminal info before using the port
48 term_info tiNew
; // Terminal info during the transaction
51 // Set time-out on 30 miliseconds
52 const struct timeval timeout
= {
53 .tv_sec
= 0, // 0 second
54 .tv_usec
= 30000 // 30000 micro seconds
57 // Work-around to claim uart interface using the c_iflag (software input processing) from the termios struct
58 #define CCLAIMED 0x80000000
60 serial_port
uart_open(const char* pcPortName
)
62 serial_port_unix
* sp
= malloc(sizeof(serial_port_unix
));
64 if (sp
== 0) return INVALID_SERIAL_PORT
;
66 sp
->fd
= open(pcPortName
, O_RDWR
| O_NOCTTY
| O_NDELAY
| O_NONBLOCK
);
70 return INVALID_SERIAL_PORT
;
73 if(tcgetattr(sp
->fd
,&sp
->tiOld
) == -1)
76 return INVALID_SERIAL_PORT
;
79 // Make sure the port is not claimed already
80 if (sp
->tiOld
.c_iflag
& CCLAIMED
)
83 return CLAIMED_SERIAL_PORT
;
86 // Copy the old terminal info struct
87 sp
->tiNew
= sp
->tiOld
;
89 sp
->tiNew
.c_cflag
= CS8
| CLOCAL
| CREAD
;
90 sp
->tiNew
.c_iflag
= CCLAIMED
| IGNPAR
;
91 sp
->tiNew
.c_oflag
= 0;
92 sp
->tiNew
.c_lflag
= 0;
94 sp
->tiNew
.c_cc
[VMIN
] = 0; // block until n bytes are received
95 sp
->tiNew
.c_cc
[VTIME
] = 0; // block until a timer expires (n * 100 mSec.)
97 if(tcsetattr(sp
->fd
,TCSANOW
,&sp
->tiNew
) == -1)
100 return INVALID_SERIAL_PORT
;
103 tcflush(sp
->fd
, TCIFLUSH
);
107 void uart_set_speed(serial_port sp
, const uint32_t uiPortSpeed
)
109 DBG("Serial port speed requested to be set to %d bauds.", uiPortSpeed
);
110 // Set port speed (Input and Output)
111 speed_t stPortSpeed
= B9600
;
112 switch(uiPortSpeed
) {
113 case 9600: stPortSpeed
= B9600
;
115 case 19200: stPortSpeed
= B19200
;
117 case 38400: stPortSpeed
= B38400
;
119 case 57600: stPortSpeed
= B57600
;
121 case 115200: stPortSpeed
= B115200
;
123 case 230400: stPortSpeed
= B230400
;
126 case 460800: stPortSpeed
= B460800
;
131 ERR("Unable to set serial port speed to %d bauds. Speed value must be one of these constants: 9600 (default), 19200, 38400, 57600, 115200, 230400 or 460800.", uiPortSpeed
);
133 ERR("Unable to set serial port speed to %d bauds. Speed value must be one of these constants: 9600 (default), 19200, 38400, 57600, 115200 or 230400.", uiPortSpeed
);
136 const serial_port_unix
* spu
= (serial_port_unix
*)sp
;
137 cfsetispeed((struct termios
*)&spu
->tiNew
, stPortSpeed
);
138 cfsetospeed((struct termios
*)&spu
->tiNew
, stPortSpeed
);
139 if( tcsetattr(spu
->fd
, TCSADRAIN
, &spu
->tiNew
) == -1)
141 ERR("Unable to apply new speed settings.");
145 uint32_t uart_get_speed(const serial_port sp
)
147 uint32_t uiPortSpeed
= 0;
148 const serial_port_unix
* spu
= (serial_port_unix
*)sp
;
149 switch (cfgetispeed(&spu
->tiNew
))
151 case B9600
: uiPortSpeed
= 9600;
153 case B19200
: uiPortSpeed
= 19200;
155 case B38400
: uiPortSpeed
= 38400;
157 case B57600
: uiPortSpeed
= 57600;
159 case B115200
: uiPortSpeed
= 115200;
161 case B230400
: uiPortSpeed
= 230400;
164 case B460800
: uiPortSpeed
= 460800;
172 void uart_close(const serial_port sp
)
174 tcsetattr(((serial_port_unix
*)sp
)->fd
,TCSANOW
,&((serial_port_unix
*)sp
)->tiOld
);
175 close(((serial_port_unix
*)sp
)->fd
);
179 bool uart_cts(const serial_port sp
)
182 if (ioctl(((serial_port_unix
*)sp
)->fd
,TIOCMGET
,&status
) < 0) return false;
183 return (status
& TIOCM_CTS
);
186 bool uart_receive(const serial_port sp
, byte_t
* pbtRx
, size_t* pszRxLen
)
193 // Reset the output count
197 // Reset file descriptor
199 FD_SET(((serial_port_unix
*)sp
)->fd
,&rfds
);
201 res
= select(((serial_port_unix
*)sp
)->fd
+1, &rfds
, NULL
, NULL
, &tv
);
211 if (*pszRxLen
== 0) {
212 // Error, we received no data
213 DBG("RX time-out, buffer empty.");
216 // We received some data, but nothing more is available
221 // Retrieve the count of the incoming bytes
222 res
= ioctl(((serial_port_unix
*)sp
)->fd
, FIONREAD
, &byteCount
);
223 if (res
< 0) return false;
225 // There is something available, read the data
226 res
= read(((serial_port_unix
*)sp
)->fd
,pbtRx
+(*pszRxLen
),byteCount
);
228 // Stop if the OS has some troubles reading the data
229 if (res
<= 0) return false;
238 bool uart_send(const serial_port sp
, const byte_t
* pbtTx
, const size_t szTxLen
)
245 while (szPos
< szTxLen
)
247 // Reset file descriptor
249 FD_SET(((serial_port_unix
*)sp
)->fd
,&rfds
);
251 res
= select(((serial_port_unix
*)sp
)->fd
+1, NULL
, &rfds
, NULL
, &tv
);
265 // Send away the bytes
266 res
= write(((serial_port_unix
*)sp
)->fd
,pbtTx
+szPos
,szTxLen
-szPos
);
268 // Stop if the OS has some troubles sending the data
269 if (res
<= 0) return false;
277 // The windows serial port implementation
280 HANDLE hPort
; // Serial port handle
281 DCB dcb
; // Device control settings
282 COMMTIMEOUTS ct
; // Serial port time-out configuration
283 } serial_port_windows
;
285 serial_port
uart_open(const char* pcPortName
)
287 char acPortName
[255];
288 serial_port_windows
* sp
= malloc(sizeof(serial_port_windows
));
290 // Copy the input "com?" to "\\.\COM?" format
291 sprintf(acPortName
,"\\\\.\\%s",pcPortName
);
294 // Try to open the serial port
295 sp
->hPort
= CreateFileA(acPortName
,GENERIC_READ
|GENERIC_WRITE
,0,NULL
,OPEN_EXISTING
,0,NULL
);
296 if (sp
->hPort
== INVALID_HANDLE_VALUE
)
299 return INVALID_SERIAL_PORT
;
302 // Prepare the device control
303 memset(&sp
->dcb
, 0, sizeof(DCB
));
304 sp
->dcb
.DCBlength
= sizeof(DCB
);
305 if(!BuildCommDCBA("baud=9600 data=8 parity=N stop=1",&sp
->dcb
))
308 return INVALID_SERIAL_PORT
;
311 // Update the active serial port
312 if(!SetCommState(sp
->hPort
,&sp
->dcb
))
315 return INVALID_SERIAL_PORT
;
318 sp
->ct
.ReadIntervalTimeout
= 0;
319 sp
->ct
.ReadTotalTimeoutMultiplier
= 0;
320 sp
->ct
.ReadTotalTimeoutConstant
= 30;
321 sp
->ct
.WriteTotalTimeoutMultiplier
= 0;
322 sp
->ct
.WriteTotalTimeoutConstant
= 30;
324 if(!SetCommTimeouts(sp
->hPort
,&sp
->ct
))
327 return INVALID_SERIAL_PORT
;
330 PurgeComm(sp
->hPort
, PURGE_RXABORT
| PURGE_RXCLEAR
);
335 void uart_close(const serial_port sp
)
337 CloseHandle(((serial_port_windows
*)sp
)->hPort
);
341 void uart_set_speed(serial_port sp
, const uint32_t uiPortSpeed
)
343 serial_port_windows
* spw
;
345 DBG("Serial port speed requested to be set to %d bauds.", uiPortSpeed
);
346 // Set port speed (Input and Output)
347 switch(uiPortSpeed
) {
357 ERR("Unable to set serial port speed to %d bauds. Speed value must be one of these constants: 9600 (default), 19200, 38400, 57600, 115200, 230400 or 460800.", uiPortSpeed
);
360 spw
= (serial_port_windows
*)sp
;
361 spw
->dcb
.BaudRate
= uiPortSpeed
;
362 if (!SetCommState(spw
->hPort
, &spw
->dcb
))
364 ERR("Unable to apply new speed settings.");
368 uint32_t uart_get_speed(const serial_port sp
)
370 const serial_port_windows
* spw
= (serial_port_windows
*)sp
;
371 if (!GetCommState(spw
->hPort
, (serial_port
)&spw
->dcb
))
372 return spw
->dcb
.BaudRate
;
377 bool uart_receive(const serial_port sp
, byte_t
* pbtRx
, size_t* pszRxLen
)
379 ReadFile(((serial_port_windows
*)sp
)->hPort
,pbtRx
,*pszRxLen
,(LPDWORD
)pszRxLen
,NULL
);
380 return (*pszRxLen
!= 0);
383 bool uart_send(const serial_port sp
, const byte_t
* pbtTx
, const size_t szTxLen
)
386 return WriteFile(((serial_port_windows
*)sp
)->hPort
,pbtTx
,szTxLen
,&dwTxLen
,NULL
);
387 return (dwTxLen
!= 0);