lapb_iface.c 9.0 KB

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