llc_macinit.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:5k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * NET An implementation of the IEEE 802.2 LLC protocol for the
  3.  * LINUX operating system.  LLC is implemented as a set of
  4.  * state machines and callbacks for higher networking layers.
  5.  *
  6.  * Code for initialization, termination, registration and
  7.  * MAC layer glue.
  8.  *
  9.  * Written by Tim Alpaerts, Tim_Alpaerts@toyota-motor-europe.com
  10.  *
  11.  * This program is free software; you can redistribute it and/or
  12.  * modify it under the terms of the GNU General Public License
  13.  * as published by the Free Software Foundation; either version
  14.  * 2 of the License, or (at your option) any later version.
  15.  *
  16.  * Changes
  17.  * Alan Cox : Chainsawed to Linux format
  18.  * Added llc_ to names
  19.  * Started restructuring handlers
  20.  *
  21.  *              Horst von Brand :      Add #include <linux/string.h>
  22.  */
  23. #include <linux/module.h>
  24. #include <linux/version.h>
  25. #include <linux/kernel.h>
  26. #include <linux/slab.h>
  27. #include <linux/unistd.h>
  28. #include <linux/string.h>
  29. #include <linux/netdevice.h>
  30. #include <linux/init.h>
  31. #include <net/p8022.h>
  32. #include <asm/byteorder.h>
  33. #include <net/llc_frame.h>
  34. #include <net/llc.h>
  35. /*
  36.  * All incoming frames pass thru mac_data_indicate().
  37.  * On entry the llc structure related to the frame is passed as parameter. 
  38.  * The received sk_buffs with pdus other than I_CMD and I_RSP
  39.  * are freed by mac_data_indicate() after processing,
  40.  * the I pdu buffers are freed by the cl2llc client when it no longer needs
  41.  * the skb.
  42. */
  43. int llc_mac_data_indicate(llcptr lp, struct sk_buff *skb)
  44. {
  45. int ll;       /* logical length == 802.3 length field */
  46. unsigned char p_flag;
  47. unsigned char type;
  48. frameptr fr;
  49. int free=1;
  50. lp->inc_skb=NULL;
  51. /*
  52.  * Truncate buffer to true 802.3 length
  53.  * [FIXME: move to 802.2 demux]
  54.  */
  55. ll = *(skb->data -2) * 256 + *(skb->data -1);
  56. skb_trim( skb, ll );
  57. fr = (frameptr) skb->data;
  58. type = llc_decode_frametype( fr );
  59. if (type <= FRMR_RSP)
  60. {
  61. /*
  62.  * PDU is of the type 2 set
  63.  */
  64. if ((lp->llc_mode == MODE_ABM)||(type == SABME_CMD))
  65. llc_process_otype2_frame(lp, skb, type);
  66. }
  67. else
  68. {
  69. /*
  70.  * PDU belongs to type 1 set
  71.  */
  72.         p_flag = fr->u_hdr.u_pflag;
  73.          switch(type)
  74.          {
  75.         case TEST_CMD:
  76. llc_sendpdu(lp, TEST_RSP, 0,ll -3,
  77. fr->u_hdr.u_info);
  78. break;
  79. case TEST_RSP:
  80. lp->llc_callbacks|=LLC_TEST_INDICATION;
  81. lp->inc_skb=skb;
  82. free=0;
  83. break;
  84. case XID_CMD:
  85. /*
  86.  * Basic format XID is handled by LLC itself
  87.  * Doc 5.4.1.1.2 p 48/49
  88.  */
  89. if ((ll == 6)&&(fr->u_hdr.u_info[0] == 0x81))
  90. {
  91. lp->k = fr->u_hdr.u_info[2];
  92. llc_sendpdu(lp, XID_RSP,
  93. fr->u_hdr.u_pflag, ll -3,
  94. fr->u_hdr.u_info);
  95. }
  96. break;
  97. case XID_RSP:
  98. if( ll == 6 && fr->u_hdr.u_info[0] == 0x81 )
  99. {
  100. lp->k = fr->u_hdr.u_info[2];
  101. }
  102. lp->llc_callbacks|=LLC_XID_INDICATION;
  103. lp->inc_skb=skb;
  104. free=0;
  105. break;
  106. case UI_CMD:
  107. lp->llc_callbacks|=LLC_UI_DATA;
  108. skb_pull(skb,3);
  109. lp->inc_skb=skb;
  110. free=0;
  111. break;
  112. default:;
  113. /*
  114.  * All other type 1 pdus ignored for now
  115.  */
  116. }
  117. }
  118. if (free&&(!(IS_IFRAME(fr))))
  119. {
  120. /*
  121.  * No auto free for I pdus
  122.  */
  123. skb->sk = NULL;
  124. kfree_skb(skb);
  125. }
  126. if(lp->llc_callbacks)
  127. {
  128. if ( lp->llc_event != NULL ) lp->llc_event(lp);
  129. lp->llc_callbacks=0;
  130. }
  131. return 0;
  132. }
  133. /*
  134.  * Create an LLC client. As it is the job of the caller to clean up
  135.  * LLC's on device down, the device list must be locked before this call.
  136.  */
  137. int register_cl2llc_client(llcptr lp, const char *device, void (*event)(llcptr), u8 *rmac, u8 ssap, u8 dsap)
  138. {
  139. char eye_init[] = "LLC";
  140. memset(lp, 0, sizeof(*lp));
  141. lp->dev = __dev_get_by_name(device);
  142. if(lp->dev == NULL)
  143. return -ENODEV;
  144. memcpy(lp->eye, eye_init, sizeof(lp->eye));
  145. lp->rw = 1;
  146. lp->k = 127;
  147. lp->n1 = 1490;
  148. lp->n2 = 10;
  149. lp->timer_interval[P_TIMER] = HZ;    /* 1 sec */
  150. lp->timer_interval[REJ_TIMER] = HZ/8;
  151. lp->timer_interval[ACK_TIMER] = HZ/8;
  152. lp->timer_interval[BUSY_TIMER] = HZ*2;
  153. lp->local_sap = ssap;
  154. lp->llc_event = event;
  155. memcpy(lp->remote_mac, rmac, sizeof(lp->remote_mac));
  156. lp->state = 0;
  157. lp->llc_mode = MODE_ADM;
  158. lp->remote_sap = dsap;
  159. skb_queue_head_init(&lp->atq);
  160. skb_queue_head_init(&lp->rtq);
  161. MOD_INC_USE_COUNT;
  162. return 0;
  163. }
  164. void unregister_cl2llc_client(llcptr lp)
  165. {
  166. llc_cancel_timers(lp);
  167. MOD_DEC_USE_COUNT;
  168. kfree(lp);
  169. }
  170. EXPORT_SYMBOL(register_cl2llc_client);
  171. EXPORT_SYMBOL(unregister_cl2llc_client);
  172. EXPORT_SYMBOL(llc_data_request);
  173. EXPORT_SYMBOL(llc_unit_data_request);
  174. EXPORT_SYMBOL(llc_test_request);
  175. EXPORT_SYMBOL(llc_xid_request);
  176. EXPORT_SYMBOL(llc_mac_data_indicate);
  177. EXPORT_SYMBOL(llc_cancel_timers);
  178. #define ALL_TYPES_8022 0
  179. static int __init llc_init(void)
  180. {
  181. printk(KERN_NOTICE "IEEE 802.2 LLC for Linux 2.1 (c) 1996 Tim Alpaertsn");
  182. return 0;
  183. }
  184. module_init(llc_init);