lapb_iface.c 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443
  1. /*
  2. * LAPB release 002
  3. *
  4. * This code REQUIRES 2.1.15 or higher/ NET3.038
  5. *
  6. * This module:
  7. * This module is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU General Public License
  9. * as published by the Free Software Foundation; either version
  10. * 2 of the License, or (at your option) any later version.
  11. *
  12. * History
  13. * LAPB 001 Jonathan Naylor Started Coding
  14. * LAPB 002 Jonathan Naylor New timer architecture.
  15. * 2000-10-29 Henner Eisen lapb_data_indication() return status.
  16. */
  17. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  18. #include <linux/module.h>
  19. #include <linux/errno.h>
  20. #include <linux/types.h>
  21. #include <linux/socket.h>
  22. #include <linux/in.h>
  23. #include <linux/kernel.h>
  24. #include <linux/jiffies.h>
  25. #include <linux/timer.h>
  26. #include <linux/string.h>
  27. #include <linux/sockios.h>
  28. #include <linux/net.h>
  29. #include <linux/inet.h>
  30. #include <linux/if_arp.h>
  31. #include <linux/skbuff.h>
  32. #include <linux/slab.h>
  33. #include <net/sock.h>
  34. #include <linux/uaccess.h>
  35. #include <linux/fcntl.h>
  36. #include <linux/mm.h>
  37. #include <linux/interrupt.h>
  38. #include <linux/stat.h>
  39. #include <linux/init.h>
  40. #include <net/lapb.h>
  41. static LIST_HEAD(lapb_list);
  42. static DEFINE_RWLOCK(lapb_list_lock);
  43. /*
  44. * Free an allocated lapb control block.
  45. */
  46. static void lapb_free_cb(struct lapb_cb *lapb)
  47. {
  48. kfree(lapb);
  49. }
  50. static __inline__ void lapb_hold(struct lapb_cb *lapb)
  51. {
  52. refcount_inc(&lapb->refcnt);
  53. }
  54. static __inline__ void lapb_put(struct lapb_cb *lapb)
  55. {
  56. if (refcount_dec_and_test(&lapb->refcnt))
  57. lapb_free_cb(lapb);
  58. }
  59. /*
  60. * Socket removal during an interrupt is now safe.
  61. */
  62. static void __lapb_remove_cb(struct lapb_cb *lapb)
  63. {
  64. if (lapb->node.next) {
  65. list_del(&lapb->node);
  66. lapb_put(lapb);
  67. }
  68. }
  69. EXPORT_SYMBOL(lapb_register);
  70. /*
  71. * Add a socket to the bound sockets list.
  72. */
  73. static void __lapb_insert_cb(struct lapb_cb *lapb)
  74. {
  75. list_add(&lapb->node, &lapb_list);
  76. lapb_hold(lapb);
  77. }
  78. static struct lapb_cb *__lapb_devtostruct(struct net_device *dev)
  79. {
  80. struct list_head *entry;
  81. struct lapb_cb *lapb, *use = NULL;
  82. list_for_each(entry, &lapb_list) {
  83. lapb = list_entry(entry, struct lapb_cb, node);
  84. if (lapb->dev == dev) {
  85. use = lapb;
  86. break;
  87. }
  88. }
  89. if (use)
  90. lapb_hold(use);
  91. return use;
  92. }
  93. static struct lapb_cb *lapb_devtostruct(struct net_device *dev)
  94. {
  95. struct lapb_cb *rc;
  96. read_lock_bh(&lapb_list_lock);
  97. rc = __lapb_devtostruct(dev);
  98. read_unlock_bh(&lapb_list_lock);
  99. return rc;
  100. }
  101. /*
  102. * Create an empty LAPB control block.
  103. */
  104. static struct lapb_cb *lapb_create_cb(void)
  105. {
  106. struct lapb_cb *lapb = kzalloc(sizeof(*lapb), GFP_ATOMIC);
  107. if (!lapb)
  108. goto out;
  109. skb_queue_head_init(&lapb->write_queue);
  110. skb_queue_head_init(&lapb->ack_queue);
  111. timer_setup(&lapb->t1timer, NULL, 0);
  112. timer_setup(&lapb->t2timer, NULL, 0);
  113. lapb->t1 = LAPB_DEFAULT_T1;
  114. lapb->t2 = LAPB_DEFAULT_T2;
  115. lapb->n2 = LAPB_DEFAULT_N2;
  116. lapb->mode = LAPB_DEFAULT_MODE;
  117. lapb->window = LAPB_DEFAULT_WINDOW;
  118. lapb->state = LAPB_STATE_0;
  119. refcount_set(&lapb->refcnt, 1);
  120. out:
  121. return lapb;
  122. }
  123. int lapb_register(struct net_device *dev,
  124. const struct lapb_register_struct *callbacks)
  125. {
  126. struct lapb_cb *lapb;
  127. int rc = LAPB_BADTOKEN;
  128. write_lock_bh(&lapb_list_lock);
  129. lapb = __lapb_devtostruct(dev);
  130. if (lapb) {
  131. lapb_put(lapb);
  132. goto out;
  133. }
  134. lapb = lapb_create_cb();
  135. rc = LAPB_NOMEM;
  136. if (!lapb)
  137. goto out;
  138. lapb->dev = dev;
  139. lapb->callbacks = callbacks;
  140. __lapb_insert_cb(lapb);
  141. lapb_start_t1timer(lapb);
  142. rc = LAPB_OK;
  143. out:
  144. write_unlock_bh(&lapb_list_lock);
  145. return rc;
  146. }
  147. int lapb_unregister(struct net_device *dev)
  148. {
  149. struct lapb_cb *lapb;
  150. int rc = LAPB_BADTOKEN;
  151. write_lock_bh(&lapb_list_lock);
  152. lapb = __lapb_devtostruct(dev);
  153. if (!lapb)
  154. goto out;
  155. lapb_put(lapb);
  156. lapb_stop_t1timer(lapb);
  157. lapb_stop_t2timer(lapb);
  158. lapb_clear_queues(lapb);
  159. __lapb_remove_cb(lapb);
  160. lapb_put(lapb);
  161. rc = LAPB_OK;
  162. out:
  163. write_unlock_bh(&lapb_list_lock);
  164. return rc;
  165. }
  166. EXPORT_SYMBOL(lapb_unregister);
  167. int lapb_getparms(struct net_device *dev, struct lapb_parms_struct *parms)
  168. {
  169. int rc = LAPB_BADTOKEN;
  170. struct lapb_cb *lapb = lapb_devtostruct(dev);
  171. if (!lapb)
  172. goto out;
  173. parms->t1 = lapb->t1 / HZ;
  174. parms->t2 = lapb->t2 / HZ;
  175. parms->n2 = lapb->n2;
  176. parms->n2count = lapb->n2count;
  177. parms->state = lapb->state;
  178. parms->window = lapb->window;
  179. parms->mode = lapb->mode;
  180. if (!timer_pending(&lapb->t1timer))
  181. parms->t1timer = 0;
  182. else
  183. parms->t1timer = (lapb->t1timer.expires - jiffies) / HZ;
  184. if (!timer_pending(&lapb->t2timer))
  185. parms->t2timer = 0;
  186. else
  187. parms->t2timer = (lapb->t2timer.expires - jiffies) / HZ;
  188. lapb_put(lapb);
  189. rc = LAPB_OK;
  190. out:
  191. return rc;
  192. }
  193. EXPORT_SYMBOL(lapb_getparms);
  194. int lapb_setparms(struct net_device *dev, struct lapb_parms_struct *parms)
  195. {
  196. int rc = LAPB_BADTOKEN;
  197. struct lapb_cb *lapb = lapb_devtostruct(dev);
  198. if (!lapb)
  199. goto out;
  200. rc = LAPB_INVALUE;
  201. if (parms->t1 < 1 || parms->t2 < 1 || parms->n2 < 1)
  202. goto out_put;
  203. if (lapb->state == LAPB_STATE_0) {
  204. if (parms->mode & LAPB_EXTENDED) {
  205. if (parms->window < 1 || parms->window > 127)
  206. goto out_put;
  207. } else {
  208. if (parms->window < 1 || parms->window > 7)
  209. goto out_put;
  210. }
  211. lapb->mode = parms->mode;
  212. lapb->window = parms->window;
  213. }
  214. lapb->t1 = parms->t1 * HZ;
  215. lapb->t2 = parms->t2 * HZ;
  216. lapb->n2 = parms->n2;
  217. rc = LAPB_OK;
  218. out_put:
  219. lapb_put(lapb);
  220. out:
  221. return rc;
  222. }
  223. EXPORT_SYMBOL(lapb_setparms);
  224. int lapb_connect_request(struct net_device *dev)
  225. {
  226. struct lapb_cb *lapb = lapb_devtostruct(dev);
  227. int rc = LAPB_BADTOKEN;
  228. if (!lapb)
  229. goto out;
  230. rc = LAPB_OK;
  231. if (lapb->state == LAPB_STATE_1)
  232. goto out_put;
  233. rc = LAPB_CONNECTED;
  234. if (lapb->state == LAPB_STATE_3 || lapb->state == LAPB_STATE_4)
  235. goto out_put;
  236. lapb_establish_data_link(lapb);
  237. lapb_dbg(0, "(%p) S0 -> S1\n", lapb->dev);
  238. lapb->state = LAPB_STATE_1;
  239. rc = LAPB_OK;
  240. out_put:
  241. lapb_put(lapb);
  242. out:
  243. return rc;
  244. }
  245. EXPORT_SYMBOL(lapb_connect_request);
  246. int lapb_disconnect_request(struct net_device *dev)
  247. {
  248. struct lapb_cb *lapb = lapb_devtostruct(dev);
  249. int rc = LAPB_BADTOKEN;
  250. if (!lapb)
  251. goto out;
  252. switch (lapb->state) {
  253. case LAPB_STATE_0:
  254. rc = LAPB_NOTCONNECTED;
  255. goto out_put;
  256. case LAPB_STATE_1:
  257. lapb_dbg(1, "(%p) S1 TX DISC(1)\n", lapb->dev);
  258. lapb_dbg(0, "(%p) S1 -> S0\n", lapb->dev);
  259. lapb_send_control(lapb, LAPB_DISC, LAPB_POLLON, LAPB_COMMAND);
  260. lapb->state = LAPB_STATE_0;
  261. lapb_start_t1timer(lapb);
  262. rc = LAPB_NOTCONNECTED;
  263. goto out_put;
  264. case LAPB_STATE_2:
  265. rc = LAPB_OK;
  266. goto out_put;
  267. }
  268. lapb_clear_queues(lapb);
  269. lapb->n2count = 0;
  270. lapb_send_control(lapb, LAPB_DISC, LAPB_POLLON, LAPB_COMMAND);
  271. lapb_start_t1timer(lapb);
  272. lapb_stop_t2timer(lapb);
  273. lapb->state = LAPB_STATE_2;
  274. lapb_dbg(1, "(%p) S3 DISC(1)\n", lapb->dev);
  275. lapb_dbg(0, "(%p) S3 -> S2\n", lapb->dev);
  276. rc = LAPB_OK;
  277. out_put:
  278. lapb_put(lapb);
  279. out:
  280. return rc;
  281. }
  282. EXPORT_SYMBOL(lapb_disconnect_request);
  283. int lapb_data_request(struct net_device *dev, struct sk_buff *skb)
  284. {
  285. struct lapb_cb *lapb = lapb_devtostruct(dev);
  286. int rc = LAPB_BADTOKEN;
  287. if (!lapb)
  288. goto out;
  289. rc = LAPB_NOTCONNECTED;
  290. if (lapb->state != LAPB_STATE_3 && lapb->state != LAPB_STATE_4)
  291. goto out_put;
  292. skb_queue_tail(&lapb->write_queue, skb);
  293. lapb_kick(lapb);
  294. rc = LAPB_OK;
  295. out_put:
  296. lapb_put(lapb);
  297. out:
  298. return rc;
  299. }
  300. EXPORT_SYMBOL(lapb_data_request);
  301. int lapb_data_received(struct net_device *dev, struct sk_buff *skb)
  302. {
  303. struct lapb_cb *lapb = lapb_devtostruct(dev);
  304. int rc = LAPB_BADTOKEN;
  305. if (lapb) {
  306. lapb_data_input(lapb, skb);
  307. lapb_put(lapb);
  308. rc = LAPB_OK;
  309. }
  310. return rc;
  311. }
  312. EXPORT_SYMBOL(lapb_data_received);
  313. void lapb_connect_confirmation(struct lapb_cb *lapb, int reason)
  314. {
  315. if (lapb->callbacks->connect_confirmation)
  316. lapb->callbacks->connect_confirmation(lapb->dev, reason);
  317. }
  318. void lapb_connect_indication(struct lapb_cb *lapb, int reason)
  319. {
  320. if (lapb->callbacks->connect_indication)
  321. lapb->callbacks->connect_indication(lapb->dev, reason);
  322. }
  323. void lapb_disconnect_confirmation(struct lapb_cb *lapb, int reason)
  324. {
  325. if (lapb->callbacks->disconnect_confirmation)
  326. lapb->callbacks->disconnect_confirmation(lapb->dev, reason);
  327. }
  328. void lapb_disconnect_indication(struct lapb_cb *lapb, int reason)
  329. {
  330. if (lapb->callbacks->disconnect_indication)
  331. lapb->callbacks->disconnect_indication(lapb->dev, reason);
  332. }
  333. int lapb_data_indication(struct lapb_cb *lapb, struct sk_buff *skb)
  334. {
  335. if (lapb->callbacks->data_indication)
  336. return lapb->callbacks->data_indication(lapb->dev, skb);
  337. kfree_skb(skb);
  338. return NET_RX_SUCCESS; /* For now; must be != NET_RX_DROP */
  339. }
  340. int lapb_data_transmit(struct lapb_cb *lapb, struct sk_buff *skb)
  341. {
  342. int used = 0;
  343. if (lapb->callbacks->data_transmit) {
  344. lapb->callbacks->data_transmit(lapb->dev, skb);
  345. used = 1;
  346. }
  347. return used;
  348. }
  349. static int __init lapb_init(void)
  350. {
  351. return 0;
  352. }
  353. static void __exit lapb_exit(void)
  354. {
  355. WARN_ON(!list_empty(&lapb_list));
  356. }
  357. MODULE_AUTHOR("Jonathan Naylor <g4klx@g4klx.demon.co.uk>");
  358. MODULE_DESCRIPTION("The X.25 Link Access Procedure B link layer protocol");
  359. MODULE_LICENSE("GPL");
  360. module_init(lapb_init);
  361. module_exit(lapb_exit);