slip_user.c 5.1 KB

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