-
Notifications
You must be signed in to change notification settings - Fork 6
/
senda_example.c
173 lines (141 loc) · 3.83 KB
/
senda_example.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
/*
* Copyright (C) 2019 Microchip Technology Inc. All rights reserved.
*
* SPDX-License-Identifier: MIT
*
* Example to exercise PARMD and SENDA 9-bit UART support on SAMA5D3.
*/
#include "custom_termios2.h"
#include <errno.h>
#include <fcntl.h>
#include <linux/serial.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ioctl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <termios.h>
#include <time.h>
#include <unistd.h>
/*
* Defined by patches https://www.spinics.net/lists/linux-serial/msg25997.html
*/
#ifndef PARMD
#warning Your kernel does not appear to support PARMD. Will try anyway.
#define PARMD 040000000
#endif
#ifndef SENDA
#warning Your kernel does not appear to support SENDA. Will try anyway.
#define SENDA 0100000000
#endif
/**
* Sends a message, but indicates the first byte should have the 9th bit set.
*/
static int send_9bit_msg(int fd, struct termios2* term, const char* buf, int size)
{
int ret;
/* enable parity, multidrop, and mark 1st byte as address */
term->c_cflag |= PARENB | CMSPAR | PARMD | SENDA;
ret = tcsetattr2(fd, TCSADRAIN, term);
if (ret < 0) {
fprintf(stderr, "tcsetattr() failed: %s\n",
strerror(errno));
return ret;
}
/* write the message */
ret = write(fd, buf, size);
if (ret < 0) {
fprintf(stderr, "write() failed: %s\n",
strerror(errno));
return ret;
}
return ret;
}
static int quit = 0;
static void signal_handler(int dummy)
{
quit = 1;
}
#define MAX_MSG_SIZE 255
int main(int argc, char** argv)
{
char buf[MAX_MSG_SIZE];
int n = 0;
int fd;
struct termios2 old_termios;
struct termios2 new_termios;
struct serial_rs485 rs485conf;
const char* dev;
int baud;
signal(SIGINT, signal_handler);
if (argc != 3) {
printf("usage: %s DEVICE BAUD", argv[0]);
return -1;
}
dev = argv[1];
baud = atoi(argv[2]);
fd = open(dev, O_RDWR | O_NOCTTY);
if (fd < 0) {
fprintf(stderr, "could not open %s\n", dev);
return -1;
}
if (tcgetattr2(fd, &old_termios)) {
fprintf(stderr, "tcgetattr() failed: %s\n",
strerror(errno));
return -1;
}
memcpy(&new_termios, &old_termios, sizeof(new_termios));
new_termios.c_cflag = CS8 | CLOCAL;
new_termios.c_cflag &= ~CBAUD;
new_termios.c_cflag |= BOTHER;
new_termios.c_ispeed = baud;
new_termios.c_ospeed = baud;
new_termios.c_iflag = 0;
new_termios.c_oflag = 0;
new_termios.c_lflag = 0;
tcflush(fd, TCIOFLUSH);
if (tcsetattr2(fd, TCSANOW, &new_termios)) {
fprintf(stderr, "tcsetattr() failed: %s\n",
strerror(errno));
return -1;
}
// https://github.com/torvalds/linux/blob/master/Documentation/serial/serial-rs485.txt
/* Enable RS485 mode: */
rs485conf.flags = SER_RS485_ENABLED;
/* Set logical level for RTS pin equal to 1 when sending: */
rs485conf.flags |= SER_RS485_RTS_ON_SEND;
/* or, set logical level for RTS pin equal to 0 when sending: */
/*rs485conf.flags &= ~(SER_RS485_RTS_ON_SEND);*/
/* Set logical level for RTS pin equal to 1 after sending: */
/*rs485conf.flags |= SER_RS485_RTS_AFTER_SEND;*/
/* or, set logical level for RTS pin equal to 0 after sending: */
rs485conf.flags &= ~(SER_RS485_RTS_AFTER_SEND);
/* Set rts delay before send, if needed: */
rs485conf.delay_rts_before_send = 0;
/* Set rts delay after send, if needed: */
rs485conf.delay_rts_after_send = 0;
/* Set this flag if you want to receive data even whilst sending data */
/*rs485conf.flags |= SER_RS485_RX_DURING_TX;*/
if (ioctl(fd, TIOCSRS485, &rs485conf)) {
fprintf(stderr, "ioctl() failed: %s\n", strerror(errno));
return -1;
}
/* build simple message */
buf[n++] = 0x1;
buf[n++] = 0x2;
buf[n++] = 0x3;
buf[n++] = 0x4;
buf[n++] = 0x5;
buf[n++] = 0x6;
while (!quit)
{
send_9bit_msg(fd, &new_termios, buf, n);
struct timespec ts = {0,0};
nanosleep(&ts, NULL);
}
/* restore termios to original */
tcsetattr2(fd, TCSANOW, &old_termios);
return 0;
}