slip_user.c 5.1 KB

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