Merge branch '2021-10-18-OF_xxx-cleanup'
[platform/kernel/u-boot.git] / tools / termios_linux.h
1 /* SPDX-License-Identifier: GPL-2.0+ */
2 /*
3  * termios fuctions to support arbitrary baudrates (on Linux)
4  *
5  * Copyright (c) 2021 Pali Rohár <pali@kernel.org>
6  * Copyright (c) 2021 Marek Behún <marek.behun@nic.cz>
7  */
8
9 #ifndef _TERMIOS_LINUX_H_
10 #define _TERMIOS_LINUX_H_
11
12 /*
13  * We need to use raw TCGETS2/TCSETS2 or TCGETS/TCSETS ioctls with the BOTHER
14  * flag in struct termios2/termios, defined in Linux headers <asm/ioctls.h>
15  * (included by <sys/ioctl.h>) and <asm/termbits.h>. Since these headers
16  * conflict with glibc's header file <termios.h>, it is not possible to use
17  * libc's termios functions and we need to reimplement them via ioctl() calls.
18  *
19  * An arbitrary baudrate is supported when the macro BOTHER is defined. The
20  * baudrate value itself is then stored into the c_ospeed and c_ispeed members.
21  * If ioctls TCGETS2/TCSETS2 are defined and supported then these fields are
22  * present in struct termios2, otherwise these fields are present in struct
23  * termios.
24  *
25  * Note that the Bnnn constants from <termios.h> need not be compatible with Bnnn
26  * constants from <asm/termbits.h>.
27  */
28
29 #include <errno.h>
30 #include <sys/ioctl.h>
31 #include <sys/types.h>
32 #include <asm/termbits.h>
33
34 #if defined(BOTHER) && defined(TCGETS2)
35 #define termios termios2
36 #endif
37
38 static inline int tcgetattr(int fd, struct termios *t)
39 {
40 #if defined(BOTHER) && defined(TCGETS2)
41         return ioctl(fd, TCGETS2, t);
42 #else
43         return ioctl(fd, TCGETS, t);
44 #endif
45 }
46
47 static inline int tcsetattr(int fd, int a, const struct termios *t)
48 {
49         int cmd;
50
51         switch (a) {
52 #if defined(BOTHER) && defined(TCGETS2)
53         case TCSANOW:
54                 cmd = TCSETS2;
55                 break;
56         case TCSADRAIN:
57                 cmd = TCSETSW2;
58                 break;
59         case TCSAFLUSH:
60                 cmd = TCSETSF2;
61                 break;
62 #else
63         case TCSANOW:
64                 cmd = TCSETS;
65                 break;
66         case TCSADRAIN:
67                 cmd = TCSETSW;
68                 break;
69         case TCSAFLUSH:
70                 cmd = TCSETSF;
71                 break;
72 #endif
73         default:
74                 errno = EINVAL;
75                 return -1;
76         }
77
78         return ioctl(fd, cmd, t);
79 }
80
81 static inline int tcdrain(int fd)
82 {
83         return ioctl(fd, TCSBRK, 1);
84 }
85
86 static inline int tcflush(int fd, int q)
87 {
88         return ioctl(fd, TCFLSH, q);
89 }
90
91 static inline int tcsendbreak(int fd, int d)
92 {
93         return ioctl(fd, TCSBRK, d);
94 }
95
96 static inline int tcflow(int fd, int a)
97 {
98         return ioctl(fd, TCXONC, a);
99 }
100
101 static inline pid_t tcgetsid(int fd)
102 {
103         pid_t sid;
104
105         if (ioctl(fd, TIOCGSID, &sid) < 0)
106                 return (pid_t)-1;
107
108         return sid;
109 }
110
111 static inline speed_t cfgetospeed(const struct termios *t)
112 {
113         return t->c_cflag & CBAUD;
114 }
115
116 static inline int cfsetospeed(struct termios *t, speed_t s)
117 {
118         if (s & ~CBAUD) {
119                 errno = EINVAL;
120                 return -1;
121         }
122
123         t->c_cflag &= ~CBAUD;
124         t->c_cflag |= s;
125
126         return 0;
127 }
128
129 #ifdef IBSHIFT
130 static inline speed_t cfgetispeed(const struct termios *t)
131 {
132         speed_t s = (t->c_cflag >> IBSHIFT) & CBAUD;
133
134         if (s == B0)
135                 return cfgetospeed(t);
136         else
137                 return s;
138 }
139
140 static inline int cfsetispeed(struct termios *t, speed_t s)
141 {
142         if (s == 0)
143                 s = B0;
144
145         if (s & ~CBAUD) {
146                 errno = EINVAL;
147                 return -1;
148         }
149
150         t->c_cflag &= ~(CBAUD << IBSHIFT);
151         t->c_cflag |= s << IBSHIFT;
152
153         return 0;
154 }
155 #else /* !IBSHIFT */
156 static inline speed_t cfgetispeed(const struct termios *t)
157 {
158         return cfgetospeed(t);
159 }
160
161 static inline int cfsetispeed(struct termios *t, speed_t s)
162 {
163         return cfsetospeed(t, s);
164 }
165 #endif /* !IBSHIFT */
166
167 static inline int cfsetspeed(struct termios *t, speed_t s)
168 {
169         if (cfsetospeed(t, s))
170                 return -1;
171 #ifdef IBSHIFT
172         if (cfsetispeed(t, s))
173                 return -1;
174 #endif
175
176         return 0;
177 }
178
179 static void cfmakeraw(struct termios *t)
180 {
181         t->c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP | INLCR | IGNCR |
182                         ICRNL | IXON);
183         t->c_oflag &= ~OPOST;
184         t->c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
185         t->c_cflag &= ~(CSIZE | PARENB);
186         t->c_cflag |= CS8;
187 }
188
189 #endif /* _TERMIOS_LINUX_H_ */