slip_user.c 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <stddef.h>
  5. #include <sched.h>
  6. #include <string.h>
  7. #include <errno.h>
  8. #include <sys/termios.h>
  9. #include <sys/wait.h>
  10. #include <sys/signal.h>
  11. #include "user_util.h"
  12. #include "kern_util.h"
  13. #include "user.h"
  14. #include "net_user.h"
  15. #include "slip.h"
  16. #include "slip_common.h"
  17. #include "helper.h"
  18. #include "os.h"
  19. void slip_user_init(void *data, void *dev)
  20. {
  21. struct slip_data *pri = data;
  22. pri->dev = dev;
  23. }
  24. static int set_up_tty(int fd)
  25. {
  26. int i;
  27. struct termios tios;
  28. if (tcgetattr(fd, &tios) < 0) {
  29. printk("could not get initial terminal attributes\n");
  30. return(-1);
  31. }
  32. tios.c_cflag = CS8 | CREAD | HUPCL | CLOCAL;
  33. tios.c_iflag = IGNBRK | IGNPAR;
  34. tios.c_oflag = 0;
  35. tios.c_lflag = 0;
  36. for (i = 0; i < NCCS; i++)
  37. tios.c_cc[i] = 0;
  38. tios.c_cc[VMIN] = 1;
  39. tios.c_cc[VTIME] = 0;
  40. cfsetospeed(&tios, B38400);
  41. cfsetispeed(&tios, B38400);
  42. if (tcsetattr(fd, TCSAFLUSH, &tios) < 0) {
  43. printk("failed to set terminal attributes\n");
  44. return(-1);
  45. }
  46. return(0);
  47. }
  48. struct slip_pre_exec_data {
  49. int stdin;
  50. int stdout;
  51. int close_me;
  52. };
  53. static void slip_pre_exec(void *arg)
  54. {
  55. struct slip_pre_exec_data *data = arg;
  56. if(data->stdin >= 0) dup2(data->stdin, 0);
  57. dup2(data->stdout, 1);
  58. if(data->close_me >= 0) os_close_file(data->close_me);
  59. }
  60. static int slip_tramp(char **argv, int fd)
  61. {
  62. struct slip_pre_exec_data pe_data;
  63. char *output;
  64. int status, pid, fds[2], err, output_len;
  65. err = os_pipe(fds, 1, 0);
  66. if(err < 0){
  67. printk("slip_tramp : pipe failed, err = %d\n", -err);
  68. goto out;
  69. }
  70. err = 0;
  71. pe_data.stdin = fd;
  72. pe_data.stdout = fds[1];
  73. pe_data.close_me = fds[0];
  74. err = run_helper(slip_pre_exec, &pe_data, argv, NULL);
  75. if(err < 0)
  76. goto out_close;
  77. pid = err;
  78. output_len = page_size();
  79. output = um_kmalloc(output_len);
  80. if(output == NULL){
  81. printk("slip_tramp : failed to allocate output buffer\n");
  82. os_kill_process(pid, 1);
  83. err = -ENOMEM;
  84. goto out_free;
  85. }
  86. os_close_file(fds[1]);
  87. read_output(fds[0], output, output_len);
  88. printk("%s", output);
  89. CATCH_EINTR(err = waitpid(pid, &status, 0));
  90. if(err < 0)
  91. err = errno;
  92. else if(!WIFEXITED(status) || (WEXITSTATUS(status) != 0)){
  93. printk("'%s' didn't exit with status 0\n", argv[0]);
  94. err = -EINVAL;
  95. }
  96. else err = 0;
  97. os_close_file(fds[0]);
  98. out_free:
  99. kfree(output);
  100. return err;
  101. out_close:
  102. os_close_file(fds[0]);
  103. os_close_file(fds[1]);
  104. out:
  105. return err;
  106. }
  107. static int slip_open(void *data)
  108. {
  109. struct slip_data *pri = data;
  110. char version_buf[sizeof("nnnnn\0")];
  111. char gate_buf[sizeof("nnn.nnn.nnn.nnn\0")];
  112. char *argv[] = { "uml_net", version_buf, "slip", "up", gate_buf,
  113. NULL };
  114. int sfd, mfd, err;
  115. err = get_pty();
  116. if(err < 0){
  117. printk("slip-open : Failed to open pty, err = %d\n", -err);
  118. goto out;
  119. }
  120. mfd = err;
  121. err = os_open_file(ptsname(mfd), of_rdwr(OPENFLAGS()), 0);
  122. if(err < 0){
  123. printk("Couldn't open tty for slip line, err = %d\n", -err);
  124. goto out_close;
  125. }
  126. sfd = err;
  127. if(set_up_tty(sfd))
  128. goto out_close2;
  129. pri->slave = sfd;
  130. pri->slip.pos = 0;
  131. pri->slip.esc = 0;
  132. if(pri->gate_addr != NULL){
  133. sprintf(version_buf, "%d", UML_NET_VERSION);
  134. strcpy(gate_buf, pri->gate_addr);
  135. err = slip_tramp(argv, sfd);
  136. if(err < 0){
  137. printk("slip_tramp failed - err = %d\n", -err);
  138. goto out_close2;
  139. }
  140. err = os_get_ifname(pri->slave, pri->name);
  141. if(err < 0){
  142. printk("get_ifname failed, err = %d\n", -err);
  143. goto out_close2;
  144. }
  145. iter_addresses(pri->dev, open_addr, pri->name);
  146. }
  147. else {
  148. err = os_set_slip(sfd);
  149. if(err < 0){
  150. printk("Failed to set slip discipline encapsulation - "
  151. "err = %d\n", -err);
  152. goto out_close2;
  153. }
  154. }
  155. return(mfd);
  156. out_close2:
  157. os_close_file(sfd);
  158. out_close:
  159. os_close_file(mfd);
  160. out:
  161. return err;
  162. }
  163. static void slip_close(int fd, void *data)
  164. {
  165. struct slip_data *pri = data;
  166. char version_buf[sizeof("nnnnn\0")];
  167. char *argv[] = { "uml_net", version_buf, "slip", "down", pri->name,
  168. NULL };
  169. int err;
  170. if(pri->gate_addr != NULL)
  171. iter_addresses(pri->dev, close_addr, pri->name);
  172. sprintf(version_buf, "%d", UML_NET_VERSION);
  173. err = slip_tramp(argv, pri->slave);
  174. if(err != 0)
  175. printk("slip_tramp failed - errno = %d\n", -err);
  176. os_close_file(fd);
  177. os_close_file(pri->slave);
  178. pri->slave = -1;
  179. }
  180. int slip_user_read(int fd, void *buf, int len, struct slip_data *pri)
  181. {
  182. return slip_proto_read(fd, buf, len, &pri->slip);
  183. }
  184. int slip_user_write(int fd, void *buf, int len, struct slip_data *pri)
  185. {
  186. return slip_proto_write(fd, buf, len, &pri->slip);
  187. }
  188. static int slip_set_mtu(int mtu, void *data)
  189. {
  190. return(mtu);
  191. }
  192. static void slip_add_addr(unsigned char *addr, unsigned char *netmask,
  193. void *data)
  194. {
  195. struct slip_data *pri = data;
  196. if(pri->slave < 0) return;
  197. open_addr(addr, netmask, pri->name);
  198. }
  199. static void slip_del_addr(unsigned char *addr, unsigned char *netmask,
  200. void *data)
  201. {
  202. struct slip_data *pri = data;
  203. if(pri->slave < 0) return;
  204. close_addr(addr, netmask, pri->name);
  205. }
  206. struct net_user_info slip_user_info = {
  207. .init = slip_user_init,
  208. .open = slip_open,
  209. .close = slip_close,
  210. .remove = NULL,
  211. .set_mtu = slip_set_mtu,
  212. .add_address = slip_add_addr,
  213. .delete_address = slip_del_addr,
  214. .max_packet = BUF_SIZE
  215. };