irlan_common.c 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227
  1. /*********************************************************************
  2. *
  3. * Filename: irlan_common.c
  4. * Version: 0.9
  5. * Description: IrDA LAN Access Protocol Implementation
  6. * Status: Experimental.
  7. * Author: Dag Brattli <dagb@cs.uit.no>
  8. * Created at: Sun Aug 31 20:14:37 1997
  9. * Modified at: Sun Dec 26 21:53:10 1999
  10. * Modified by: Dag Brattli <dagb@cs.uit.no>
  11. *
  12. * Copyright (c) 1997, 1999 Dag Brattli <dagb@cs.uit.no>,
  13. * All Rights Reserved.
  14. *
  15. * This program is free software; you can redistribute it and/or
  16. * modify it under the terms of the GNU General Public License as
  17. * published by the Free Software Foundation; either version 2 of
  18. * the License, or (at your option) any later version.
  19. *
  20. * Neither Dag Brattli nor University of Tromsø admit liability nor
  21. * provide warranty for any of this software. This material is
  22. * provided "AS-IS" and at no charge.
  23. *
  24. ********************************************************************/
  25. #include <linux/module.h>
  26. #include <linux/kernel.h>
  27. #include <linux/string.h>
  28. #include <linux/init.h>
  29. #include <linux/errno.h>
  30. #include <linux/proc_fs.h>
  31. #include <linux/seq_file.h>
  32. #include <linux/random.h>
  33. #include <linux/netdevice.h>
  34. #include <linux/etherdevice.h>
  35. #include <linux/rtnetlink.h>
  36. #include <linux/moduleparam.h>
  37. #include <linux/bitops.h>
  38. #include <asm/system.h>
  39. #include <asm/byteorder.h>
  40. #include <net/irda/irda.h>
  41. #include <net/irda/irttp.h>
  42. #include <net/irda/irlmp.h>
  43. #include <net/irda/iriap.h>
  44. #include <net/irda/timer.h>
  45. #include <net/irda/irlan_common.h>
  46. #include <net/irda/irlan_client.h>
  47. #include <net/irda/irlan_provider.h>
  48. #include <net/irda/irlan_eth.h>
  49. #include <net/irda/irlan_filter.h>
  50. /*
  51. * Send gratuitous ARP when connected to a new AP or not. May be a clever
  52. * thing to do, but for some reason the machine crashes if you use DHCP. So
  53. * lets not use it by default.
  54. */
  55. #undef CONFIG_IRLAN_SEND_GRATUITOUS_ARP
  56. /* extern char sysctl_devname[]; */
  57. /*
  58. * Master structure
  59. */
  60. static LIST_HEAD(irlans);
  61. static void *ckey;
  62. static void *skey;
  63. /* Module parameters */
  64. static int eth; /* Use "eth" or "irlan" name for devices */
  65. static int access = ACCESS_PEER; /* PEER, DIRECT or HOSTED */
  66. #ifdef CONFIG_PROC_FS
  67. static const char *irlan_access[] = {
  68. "UNKNOWN",
  69. "DIRECT",
  70. "PEER",
  71. "HOSTED"
  72. };
  73. static const char *irlan_media[] = {
  74. "UNKNOWN",
  75. "802.3",
  76. "802.5"
  77. };
  78. extern struct proc_dir_entry *proc_irda;
  79. static int irlan_seq_open(struct inode *inode, struct file *file);
  80. static struct file_operations irlan_fops = {
  81. .owner = THIS_MODULE,
  82. .open = irlan_seq_open,
  83. .read = seq_read,
  84. .llseek = seq_lseek,
  85. .release = seq_release,
  86. };
  87. extern struct proc_dir_entry *proc_irda;
  88. #endif /* CONFIG_PROC_FS */
  89. static struct irlan_cb *irlan_open(__u32 saddr, __u32 daddr);
  90. static void __irlan_close(struct irlan_cb *self);
  91. static int __irlan_insert_param(struct sk_buff *skb, char *param, int type,
  92. __u8 value_byte, __u16 value_short,
  93. __u8 *value_array, __u16 value_len);
  94. static void irlan_open_unicast_addr(struct irlan_cb *self);
  95. static void irlan_get_unicast_addr(struct irlan_cb *self);
  96. void irlan_close_tsaps(struct irlan_cb *self);
  97. /*
  98. * Function irlan_init (void)
  99. *
  100. * Initialize IrLAN layer
  101. *
  102. */
  103. static int __init irlan_init(void)
  104. {
  105. struct irlan_cb *new;
  106. __u16 hints;
  107. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  108. #ifdef CONFIG_PROC_FS
  109. { struct proc_dir_entry *proc;
  110. proc = create_proc_entry("irlan", 0, proc_irda);
  111. if (!proc) {
  112. printk(KERN_ERR "irlan_init: can't create /proc entry!\n");
  113. return -ENODEV;
  114. }
  115. proc->proc_fops = &irlan_fops;
  116. }
  117. #endif /* CONFIG_PROC_FS */
  118. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  119. hints = irlmp_service_to_hint(S_LAN);
  120. /* Register with IrLMP as a client */
  121. ckey = irlmp_register_client(hints, &irlan_client_discovery_indication,
  122. NULL, NULL);
  123. /* Register with IrLMP as a service */
  124. skey = irlmp_register_service(hints);
  125. /* Start the master IrLAN instance (the only one for now) */
  126. new = irlan_open(DEV_ADDR_ANY, DEV_ADDR_ANY);
  127. /* The master will only open its (listen) control TSAP */
  128. irlan_provider_open_ctrl_tsap(new);
  129. /* Do some fast discovery! */
  130. irlmp_discovery_request(DISCOVERY_DEFAULT_SLOTS);
  131. return 0;
  132. }
  133. static void __exit irlan_cleanup(void)
  134. {
  135. struct irlan_cb *self, *next;
  136. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  137. irlmp_unregister_client(ckey);
  138. irlmp_unregister_service(skey);
  139. #ifdef CONFIG_PROC_FS
  140. remove_proc_entry("irlan", proc_irda);
  141. #endif /* CONFIG_PROC_FS */
  142. /* Cleanup any leftover network devices */
  143. rtnl_lock();
  144. list_for_each_entry_safe(self, next, &irlans, dev_list) {
  145. __irlan_close(self);
  146. }
  147. rtnl_unlock();
  148. }
  149. /*
  150. * Function irlan_open (void)
  151. *
  152. * Open new instance of a client/provider, we should only register the
  153. * network device if this instance is ment for a particular client/provider
  154. */
  155. static struct irlan_cb *irlan_open(__u32 saddr, __u32 daddr)
  156. {
  157. struct net_device *dev;
  158. struct irlan_cb *self;
  159. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  160. /* Create network device with irlan */
  161. dev = alloc_irlandev(eth ? "eth%d" : "irlan%d");
  162. if (!dev)
  163. return NULL;
  164. self = dev->priv;
  165. self->dev = dev;
  166. /*
  167. * Initialize local device structure
  168. */
  169. self->magic = IRLAN_MAGIC;
  170. self->saddr = saddr;
  171. self->daddr = daddr;
  172. /* Provider access can only be PEER, DIRECT, or HOSTED */
  173. self->provider.access_type = access;
  174. if (access == ACCESS_DIRECT) {
  175. /*
  176. * Since we are emulating an IrLAN sever we will have to
  177. * give ourself an ethernet address!
  178. */
  179. dev->dev_addr[0] = 0x40;
  180. dev->dev_addr[1] = 0x00;
  181. dev->dev_addr[2] = 0x00;
  182. dev->dev_addr[3] = 0x00;
  183. get_random_bytes(dev->dev_addr+4, 1);
  184. get_random_bytes(dev->dev_addr+5, 1);
  185. }
  186. self->media = MEDIA_802_3;
  187. self->disconnect_reason = LM_USER_REQUEST;
  188. init_timer(&self->watchdog_timer);
  189. init_timer(&self->client.kick_timer);
  190. init_waitqueue_head(&self->open_wait);
  191. skb_queue_head_init(&self->client.txq);
  192. irlan_next_client_state(self, IRLAN_IDLE);
  193. irlan_next_provider_state(self, IRLAN_IDLE);
  194. if (register_netdev(dev)) {
  195. IRDA_DEBUG(2, "%s(), register_netdev() failed!\n",
  196. __FUNCTION__ );
  197. self = NULL;
  198. free_netdev(dev);
  199. } else {
  200. rtnl_lock();
  201. list_add_rcu(&self->dev_list, &irlans);
  202. rtnl_unlock();
  203. }
  204. return self;
  205. }
  206. /*
  207. * Function __irlan_close (self)
  208. *
  209. * This function closes and deallocates the IrLAN client instances. Be
  210. * aware that other functions which calls client_close() must
  211. * remove self from irlans list first.
  212. */
  213. static void __irlan_close(struct irlan_cb *self)
  214. {
  215. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  216. ASSERT_RTNL();
  217. IRDA_ASSERT(self != NULL, return;);
  218. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  219. del_timer_sync(&self->watchdog_timer);
  220. del_timer_sync(&self->client.kick_timer);
  221. /* Close all open connections and remove TSAPs */
  222. irlan_close_tsaps(self);
  223. if (self->client.iriap)
  224. iriap_close(self->client.iriap);
  225. /* Remove frames queued on the control channel */
  226. skb_queue_purge(&self->client.txq);
  227. /* Unregister and free self via destructor */
  228. unregister_netdevice(self->dev);
  229. }
  230. /* Find any instance of irlan, used for client discovery wakeup */
  231. struct irlan_cb *irlan_get_any(void)
  232. {
  233. struct irlan_cb *self;
  234. list_for_each_entry_rcu(self, &irlans, dev_list) {
  235. return self;
  236. }
  237. return NULL;
  238. }
  239. /*
  240. * Function irlan_connect_indication (instance, sap, qos, max_sdu_size, skb)
  241. *
  242. * Here we receive the connect indication for the data channel
  243. *
  244. */
  245. static void irlan_connect_indication(void *instance, void *sap,
  246. struct qos_info *qos,
  247. __u32 max_sdu_size,
  248. __u8 max_header_size,
  249. struct sk_buff *skb)
  250. {
  251. struct irlan_cb *self;
  252. struct tsap_cb *tsap;
  253. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  254. self = (struct irlan_cb *) instance;
  255. tsap = (struct tsap_cb *) sap;
  256. IRDA_ASSERT(self != NULL, return;);
  257. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  258. IRDA_ASSERT(tsap == self->tsap_data,return;);
  259. self->max_sdu_size = max_sdu_size;
  260. self->max_header_size = max_header_size;
  261. IRDA_DEBUG(0, "%s: We are now connected!\n", __FUNCTION__);
  262. del_timer(&self->watchdog_timer);
  263. /* If you want to pass the skb to *both* state machines, you will
  264. * need to skb_clone() it, so that you don't free it twice.
  265. * As the state machines don't need it, git rid of it here...
  266. * Jean II */
  267. if (skb)
  268. dev_kfree_skb(skb);
  269. irlan_do_provider_event(self, IRLAN_DATA_CONNECT_INDICATION, NULL);
  270. irlan_do_client_event(self, IRLAN_DATA_CONNECT_INDICATION, NULL);
  271. if (self->provider.access_type == ACCESS_PEER) {
  272. /*
  273. * Data channel is open, so we are now allowed to
  274. * configure the remote filter
  275. */
  276. irlan_get_unicast_addr(self);
  277. irlan_open_unicast_addr(self);
  278. }
  279. /* Ready to transfer Ethernet frames (at last) */
  280. netif_start_queue(self->dev); /* Clear reason */
  281. }
  282. static void irlan_connect_confirm(void *instance, void *sap,
  283. struct qos_info *qos,
  284. __u32 max_sdu_size,
  285. __u8 max_header_size,
  286. struct sk_buff *skb)
  287. {
  288. struct irlan_cb *self;
  289. self = (struct irlan_cb *) instance;
  290. IRDA_ASSERT(self != NULL, return;);
  291. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  292. self->max_sdu_size = max_sdu_size;
  293. self->max_header_size = max_header_size;
  294. /* TODO: we could set the MTU depending on the max_sdu_size */
  295. IRDA_DEBUG(0, "%s: We are now connected!\n", __FUNCTION__);
  296. del_timer(&self->watchdog_timer);
  297. /*
  298. * Data channel is open, so we are now allowed to configure the remote
  299. * filter
  300. */
  301. irlan_get_unicast_addr(self);
  302. irlan_open_unicast_addr(self);
  303. /* Open broadcast and multicast filter by default */
  304. irlan_set_broadcast_filter(self, TRUE);
  305. irlan_set_multicast_filter(self, TRUE);
  306. /* Ready to transfer Ethernet frames */
  307. netif_start_queue(self->dev);
  308. self->disconnect_reason = 0; /* Clear reason */
  309. #ifdef CONFIG_IRLAN_SEND_GRATUITOUS_ARP
  310. irlan_eth_send_gratuitous_arp(&self->dev);
  311. #endif
  312. wake_up_interruptible(&self->open_wait);
  313. }
  314. /*
  315. * Function irlan_client_disconnect_indication (handle)
  316. *
  317. * Callback function for the IrTTP layer. Indicates a disconnection of
  318. * the specified connection (handle)
  319. */
  320. static void irlan_disconnect_indication(void *instance,
  321. void *sap, LM_REASON reason,
  322. struct sk_buff *userdata)
  323. {
  324. struct irlan_cb *self;
  325. struct tsap_cb *tsap;
  326. IRDA_DEBUG(0, "%s(), reason=%d\n", __FUNCTION__ , reason);
  327. self = (struct irlan_cb *) instance;
  328. tsap = (struct tsap_cb *) sap;
  329. IRDA_ASSERT(self != NULL, return;);
  330. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  331. IRDA_ASSERT(tsap != NULL, return;);
  332. IRDA_ASSERT(tsap->magic == TTP_TSAP_MAGIC, return;);
  333. IRDA_ASSERT(tsap == self->tsap_data, return;);
  334. IRDA_DEBUG(2, "IrLAN, data channel disconnected by peer!\n");
  335. /* Save reason so we know if we should try to reconnect or not */
  336. self->disconnect_reason = reason;
  337. switch (reason) {
  338. case LM_USER_REQUEST: /* User request */
  339. IRDA_DEBUG(2, "%s(), User requested\n", __FUNCTION__ );
  340. break;
  341. case LM_LAP_DISCONNECT: /* Unexpected IrLAP disconnect */
  342. IRDA_DEBUG(2, "%s(), Unexpected IrLAP disconnect\n", __FUNCTION__ );
  343. break;
  344. case LM_CONNECT_FAILURE: /* Failed to establish IrLAP connection */
  345. IRDA_DEBUG(2, "%s(), IrLAP connect failed\n", __FUNCTION__ );
  346. break;
  347. case LM_LAP_RESET: /* IrLAP reset */
  348. IRDA_DEBUG(2, "%s(), IrLAP reset\n", __FUNCTION__ );
  349. break;
  350. case LM_INIT_DISCONNECT:
  351. IRDA_DEBUG(2, "%s(), IrLMP connect failed\n", __FUNCTION__ );
  352. break;
  353. default:
  354. IRDA_ERROR("%s(), Unknown disconnect reason\n", __FUNCTION__);
  355. break;
  356. }
  357. /* If you want to pass the skb to *both* state machines, you will
  358. * need to skb_clone() it, so that you don't free it twice.
  359. * As the state machines don't need it, git rid of it here...
  360. * Jean II */
  361. if (userdata)
  362. dev_kfree_skb(userdata);
  363. irlan_do_client_event(self, IRLAN_LMP_DISCONNECT, NULL);
  364. irlan_do_provider_event(self, IRLAN_LMP_DISCONNECT, NULL);
  365. wake_up_interruptible(&self->open_wait);
  366. }
  367. void irlan_open_data_tsap(struct irlan_cb *self)
  368. {
  369. struct tsap_cb *tsap;
  370. notify_t notify;
  371. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  372. IRDA_ASSERT(self != NULL, return;);
  373. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  374. /* Check if already open */
  375. if (self->tsap_data)
  376. return;
  377. irda_notify_init(&notify);
  378. notify.data_indication = irlan_eth_receive;
  379. notify.udata_indication = irlan_eth_receive;
  380. notify.connect_indication = irlan_connect_indication;
  381. notify.connect_confirm = irlan_connect_confirm;
  382. notify.flow_indication = irlan_eth_flow_indication;
  383. notify.disconnect_indication = irlan_disconnect_indication;
  384. notify.instance = self;
  385. strlcpy(notify.name, "IrLAN data", sizeof(notify.name));
  386. tsap = irttp_open_tsap(LSAP_ANY, DEFAULT_INITIAL_CREDIT, &notify);
  387. if (!tsap) {
  388. IRDA_DEBUG(2, "%s(), Got no tsap!\n", __FUNCTION__ );
  389. return;
  390. }
  391. self->tsap_data = tsap;
  392. /*
  393. * This is the data TSAP selector which we will pass to the client
  394. * when the client ask for it.
  395. */
  396. self->stsap_sel_data = self->tsap_data->stsap_sel;
  397. }
  398. void irlan_close_tsaps(struct irlan_cb *self)
  399. {
  400. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  401. IRDA_ASSERT(self != NULL, return;);
  402. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  403. /* Disconnect and close all open TSAP connections */
  404. if (self->tsap_data) {
  405. irttp_disconnect_request(self->tsap_data, NULL, P_NORMAL);
  406. irttp_close_tsap(self->tsap_data);
  407. self->tsap_data = NULL;
  408. }
  409. if (self->client.tsap_ctrl) {
  410. irttp_disconnect_request(self->client.tsap_ctrl, NULL,
  411. P_NORMAL);
  412. irttp_close_tsap(self->client.tsap_ctrl);
  413. self->client.tsap_ctrl = NULL;
  414. }
  415. if (self->provider.tsap_ctrl) {
  416. irttp_disconnect_request(self->provider.tsap_ctrl, NULL,
  417. P_NORMAL);
  418. irttp_close_tsap(self->provider.tsap_ctrl);
  419. self->provider.tsap_ctrl = NULL;
  420. }
  421. self->disconnect_reason = LM_USER_REQUEST;
  422. }
  423. /*
  424. * Function irlan_ias_register (self, tsap_sel)
  425. *
  426. * Register with LM-IAS
  427. *
  428. */
  429. void irlan_ias_register(struct irlan_cb *self, __u8 tsap_sel)
  430. {
  431. struct ias_object *obj;
  432. struct ias_value *new_value;
  433. IRDA_ASSERT(self != NULL, return;);
  434. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  435. /*
  436. * Check if object has already been registered by a previous provider.
  437. * If that is the case, we just change the value of the attribute
  438. */
  439. if (!irias_find_object("IrLAN")) {
  440. obj = irias_new_object("IrLAN", IAS_IRLAN_ID);
  441. irias_add_integer_attrib(obj, "IrDA:TinyTP:LsapSel", tsap_sel,
  442. IAS_KERNEL_ATTR);
  443. irias_insert_object(obj);
  444. } else {
  445. new_value = irias_new_integer_value(tsap_sel);
  446. irias_object_change_attribute("IrLAN", "IrDA:TinyTP:LsapSel",
  447. new_value);
  448. }
  449. /* Register PnP object only if not registered before */
  450. if (!irias_find_object("PnP")) {
  451. obj = irias_new_object("PnP", IAS_PNP_ID);
  452. #if 0
  453. irias_add_string_attrib(obj, "Name", sysctl_devname,
  454. IAS_KERNEL_ATTR);
  455. #else
  456. irias_add_string_attrib(obj, "Name", "Linux", IAS_KERNEL_ATTR);
  457. #endif
  458. irias_add_string_attrib(obj, "DeviceID", "HWP19F0",
  459. IAS_KERNEL_ATTR);
  460. irias_add_integer_attrib(obj, "CompCnt", 1, IAS_KERNEL_ATTR);
  461. if (self->provider.access_type == ACCESS_PEER)
  462. irias_add_string_attrib(obj, "Comp#01", "PNP8389",
  463. IAS_KERNEL_ATTR);
  464. else
  465. irias_add_string_attrib(obj, "Comp#01", "PNP8294",
  466. IAS_KERNEL_ATTR);
  467. irias_add_string_attrib(obj, "Manufacturer",
  468. "Linux-IrDA Project", IAS_KERNEL_ATTR);
  469. irias_insert_object(obj);
  470. }
  471. }
  472. /*
  473. * Function irlan_run_ctrl_tx_queue (self)
  474. *
  475. * Try to send the next command in the control transmit queue
  476. *
  477. */
  478. int irlan_run_ctrl_tx_queue(struct irlan_cb *self)
  479. {
  480. struct sk_buff *skb;
  481. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  482. if (irda_lock(&self->client.tx_busy) == FALSE)
  483. return -EBUSY;
  484. skb = skb_dequeue(&self->client.txq);
  485. if (!skb) {
  486. self->client.tx_busy = FALSE;
  487. return 0;
  488. }
  489. /* Check that it's really possible to send commands */
  490. if ((self->client.tsap_ctrl == NULL) ||
  491. (self->client.state == IRLAN_IDLE))
  492. {
  493. self->client.tx_busy = FALSE;
  494. dev_kfree_skb(skb);
  495. return -1;
  496. }
  497. IRDA_DEBUG(2, "%s(), sending ...\n", __FUNCTION__ );
  498. return irttp_data_request(self->client.tsap_ctrl, skb);
  499. }
  500. /*
  501. * Function irlan_ctrl_data_request (self, skb)
  502. *
  503. * This function makes sure that commands on the control channel is being
  504. * sent in a command/response fashion
  505. */
  506. static void irlan_ctrl_data_request(struct irlan_cb *self, struct sk_buff *skb)
  507. {
  508. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  509. /* Queue command */
  510. skb_queue_tail(&self->client.txq, skb);
  511. /* Try to send command */
  512. irlan_run_ctrl_tx_queue(self);
  513. }
  514. /*
  515. * Function irlan_get_provider_info (self)
  516. *
  517. * Send Get Provider Information command to peer IrLAN layer
  518. *
  519. */
  520. void irlan_get_provider_info(struct irlan_cb *self)
  521. {
  522. struct sk_buff *skb;
  523. __u8 *frame;
  524. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  525. IRDA_ASSERT(self != NULL, return;);
  526. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  527. skb = alloc_skb(IRLAN_MAX_HEADER + IRLAN_CMD_HEADER,
  528. GFP_ATOMIC);
  529. if (!skb)
  530. return;
  531. /* Reserve space for TTP, LMP, and LAP header */
  532. skb_reserve(skb, self->client.max_header_size);
  533. skb_put(skb, 2);
  534. frame = skb->data;
  535. frame[0] = CMD_GET_PROVIDER_INFO;
  536. frame[1] = 0x00; /* Zero parameters */
  537. irlan_ctrl_data_request(self, skb);
  538. }
  539. /*
  540. * Function irlan_open_data_channel (self)
  541. *
  542. * Send an Open Data Command to provider
  543. *
  544. */
  545. void irlan_open_data_channel(struct irlan_cb *self)
  546. {
  547. struct sk_buff *skb;
  548. __u8 *frame;
  549. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  550. IRDA_ASSERT(self != NULL, return;);
  551. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  552. skb = alloc_skb(IRLAN_MAX_HEADER + IRLAN_CMD_HEADER +
  553. IRLAN_STRING_PARAMETER_LEN("MEDIA", "802.3") +
  554. IRLAN_STRING_PARAMETER_LEN("ACCESS_TYPE", "DIRECT"),
  555. GFP_ATOMIC);
  556. if (!skb)
  557. return;
  558. skb_reserve(skb, self->client.max_header_size);
  559. skb_put(skb, 2);
  560. frame = skb->data;
  561. /* Build frame */
  562. frame[0] = CMD_OPEN_DATA_CHANNEL;
  563. frame[1] = 0x02; /* Two parameters */
  564. irlan_insert_string_param(skb, "MEDIA", "802.3");
  565. irlan_insert_string_param(skb, "ACCESS_TYPE", "DIRECT");
  566. /* irlan_insert_string_param(skb, "MODE", "UNRELIABLE"); */
  567. /* self->use_udata = TRUE; */
  568. irlan_ctrl_data_request(self, skb);
  569. }
  570. void irlan_close_data_channel(struct irlan_cb *self)
  571. {
  572. struct sk_buff *skb;
  573. __u8 *frame;
  574. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  575. IRDA_ASSERT(self != NULL, return;);
  576. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  577. /* Check if the TSAP is still there */
  578. if (self->client.tsap_ctrl == NULL)
  579. return;
  580. skb = alloc_skb(IRLAN_MAX_HEADER + IRLAN_CMD_HEADER +
  581. IRLAN_BYTE_PARAMETER_LEN("DATA_CHAN"),
  582. GFP_ATOMIC);
  583. if (!skb)
  584. return;
  585. skb_reserve(skb, self->client.max_header_size);
  586. skb_put(skb, 2);
  587. frame = skb->data;
  588. /* Build frame */
  589. frame[0] = CMD_CLOSE_DATA_CHAN;
  590. frame[1] = 0x01; /* One parameter */
  591. irlan_insert_byte_param(skb, "DATA_CHAN", self->dtsap_sel_data);
  592. irlan_ctrl_data_request(self, skb);
  593. }
  594. /*
  595. * Function irlan_open_unicast_addr (self)
  596. *
  597. * Make IrLAN provider accept ethernet frames addressed to the unicast
  598. * address.
  599. *
  600. */
  601. static void irlan_open_unicast_addr(struct irlan_cb *self)
  602. {
  603. struct sk_buff *skb;
  604. __u8 *frame;
  605. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  606. IRDA_ASSERT(self != NULL, return;);
  607. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  608. skb = alloc_skb(IRLAN_MAX_HEADER + IRLAN_CMD_HEADER +
  609. IRLAN_BYTE_PARAMETER_LEN("DATA_CHAN") +
  610. IRLAN_STRING_PARAMETER_LEN("FILTER_TYPE", "DIRECTED") +
  611. IRLAN_STRING_PARAMETER_LEN("FILTER_MODE", "FILTER"),
  612. GFP_ATOMIC);
  613. if (!skb)
  614. return;
  615. /* Reserve space for TTP, LMP, and LAP header */
  616. skb_reserve(skb, self->max_header_size);
  617. skb_put(skb, 2);
  618. frame = skb->data;
  619. frame[0] = CMD_FILTER_OPERATION;
  620. frame[1] = 0x03; /* Three parameters */
  621. irlan_insert_byte_param(skb, "DATA_CHAN" , self->dtsap_sel_data);
  622. irlan_insert_string_param(skb, "FILTER_TYPE", "DIRECTED");
  623. irlan_insert_string_param(skb, "FILTER_MODE", "FILTER");
  624. irlan_ctrl_data_request(self, skb);
  625. }
  626. /*
  627. * Function irlan_set_broadcast_filter (self, status)
  628. *
  629. * Make IrLAN provider accept ethernet frames addressed to the broadcast
  630. * address. Be careful with the use of this one, since there may be a lot
  631. * of broadcast traffic out there. We can still function without this
  632. * one but then _we_ have to initiate all communication with other
  633. * hosts, since ARP request for this host will not be answered.
  634. */
  635. void irlan_set_broadcast_filter(struct irlan_cb *self, int status)
  636. {
  637. struct sk_buff *skb;
  638. __u8 *frame;
  639. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  640. IRDA_ASSERT(self != NULL, return;);
  641. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  642. skb = alloc_skb(IRLAN_MAX_HEADER + IRLAN_CMD_HEADER +
  643. IRLAN_BYTE_PARAMETER_LEN("DATA_CHAN") +
  644. IRLAN_STRING_PARAMETER_LEN("FILTER_TYPE", "BROADCAST") +
  645. /* We may waste one byte here...*/
  646. IRLAN_STRING_PARAMETER_LEN("FILTER_MODE", "FILTER"),
  647. GFP_ATOMIC);
  648. if (!skb)
  649. return;
  650. /* Reserve space for TTP, LMP, and LAP header */
  651. skb_reserve(skb, self->client.max_header_size);
  652. skb_put(skb, 2);
  653. frame = skb->data;
  654. frame[0] = CMD_FILTER_OPERATION;
  655. frame[1] = 0x03; /* Three parameters */
  656. irlan_insert_byte_param(skb, "DATA_CHAN", self->dtsap_sel_data);
  657. irlan_insert_string_param(skb, "FILTER_TYPE", "BROADCAST");
  658. if (status)
  659. irlan_insert_string_param(skb, "FILTER_MODE", "FILTER");
  660. else
  661. irlan_insert_string_param(skb, "FILTER_MODE", "NONE");
  662. irlan_ctrl_data_request(self, skb);
  663. }
  664. /*
  665. * Function irlan_set_multicast_filter (self, status)
  666. *
  667. * Make IrLAN provider accept ethernet frames addressed to the multicast
  668. * address.
  669. *
  670. */
  671. void irlan_set_multicast_filter(struct irlan_cb *self, int status)
  672. {
  673. struct sk_buff *skb;
  674. __u8 *frame;
  675. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  676. IRDA_ASSERT(self != NULL, return;);
  677. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  678. skb = alloc_skb(IRLAN_MAX_HEADER + IRLAN_CMD_HEADER +
  679. IRLAN_BYTE_PARAMETER_LEN("DATA_CHAN") +
  680. IRLAN_STRING_PARAMETER_LEN("FILTER_TYPE", "MULTICAST") +
  681. /* We may waste one byte here...*/
  682. IRLAN_STRING_PARAMETER_LEN("FILTER_MODE", "NONE"),
  683. GFP_ATOMIC);
  684. if (!skb)
  685. return;
  686. /* Reserve space for TTP, LMP, and LAP header */
  687. skb_reserve(skb, self->client.max_header_size);
  688. skb_put(skb, 2);
  689. frame = skb->data;
  690. frame[0] = CMD_FILTER_OPERATION;
  691. frame[1] = 0x03; /* Three parameters */
  692. irlan_insert_byte_param(skb, "DATA_CHAN", self->dtsap_sel_data);
  693. irlan_insert_string_param(skb, "FILTER_TYPE", "MULTICAST");
  694. if (status)
  695. irlan_insert_string_param(skb, "FILTER_MODE", "ALL");
  696. else
  697. irlan_insert_string_param(skb, "FILTER_MODE", "NONE");
  698. irlan_ctrl_data_request(self, skb);
  699. }
  700. /*
  701. * Function irlan_get_unicast_addr (self)
  702. *
  703. * Retrieves the unicast address from the IrLAN provider. This address
  704. * will be inserted into the devices structure, so the ethernet layer
  705. * can construct its packets.
  706. *
  707. */
  708. static void irlan_get_unicast_addr(struct irlan_cb *self)
  709. {
  710. struct sk_buff *skb;
  711. __u8 *frame;
  712. IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );
  713. IRDA_ASSERT(self != NULL, return;);
  714. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  715. skb = alloc_skb(IRLAN_MAX_HEADER + IRLAN_CMD_HEADER +
  716. IRLAN_BYTE_PARAMETER_LEN("DATA_CHAN") +
  717. IRLAN_STRING_PARAMETER_LEN("FILTER_TYPE", "DIRECTED") +
  718. IRLAN_STRING_PARAMETER_LEN("FILTER_OPERATION",
  719. "DYNAMIC"),
  720. GFP_ATOMIC);
  721. if (!skb)
  722. return;
  723. /* Reserve space for TTP, LMP, and LAP header */
  724. skb_reserve(skb, self->client.max_header_size);
  725. skb_put(skb, 2);
  726. frame = skb->data;
  727. frame[0] = CMD_FILTER_OPERATION;
  728. frame[1] = 0x03; /* Three parameters */
  729. irlan_insert_byte_param(skb, "DATA_CHAN", self->dtsap_sel_data);
  730. irlan_insert_string_param(skb, "FILTER_TYPE", "DIRECTED");
  731. irlan_insert_string_param(skb, "FILTER_OPERATION", "DYNAMIC");
  732. irlan_ctrl_data_request(self, skb);
  733. }
  734. /*
  735. * Function irlan_get_media_char (self)
  736. *
  737. *
  738. *
  739. */
  740. void irlan_get_media_char(struct irlan_cb *self)
  741. {
  742. struct sk_buff *skb;
  743. __u8 *frame;
  744. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  745. IRDA_ASSERT(self != NULL, return;);
  746. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return;);
  747. skb = alloc_skb(IRLAN_MAX_HEADER + IRLAN_CMD_HEADER +
  748. IRLAN_STRING_PARAMETER_LEN("MEDIA", "802.3"),
  749. GFP_ATOMIC);
  750. if (!skb)
  751. return;
  752. /* Reserve space for TTP, LMP, and LAP header */
  753. skb_reserve(skb, self->client.max_header_size);
  754. skb_put(skb, 2);
  755. frame = skb->data;
  756. /* Build frame */
  757. frame[0] = CMD_GET_MEDIA_CHAR;
  758. frame[1] = 0x01; /* One parameter */
  759. irlan_insert_string_param(skb, "MEDIA", "802.3");
  760. irlan_ctrl_data_request(self, skb);
  761. }
  762. /*
  763. * Function insert_byte_param (skb, param, value)
  764. *
  765. * Insert byte parameter into frame
  766. *
  767. */
  768. int irlan_insert_byte_param(struct sk_buff *skb, char *param, __u8 value)
  769. {
  770. return __irlan_insert_param(skb, param, IRLAN_BYTE, value, 0, NULL, 0);
  771. }
  772. int irlan_insert_short_param(struct sk_buff *skb, char *param, __u16 value)
  773. {
  774. return __irlan_insert_param(skb, param, IRLAN_SHORT, 0, value, NULL, 0);
  775. }
  776. /*
  777. * Function insert_string (skb, param, value)
  778. *
  779. * Insert string parameter into frame
  780. *
  781. */
  782. int irlan_insert_string_param(struct sk_buff *skb, char *param, char *string)
  783. {
  784. int string_len = strlen(string);
  785. return __irlan_insert_param(skb, param, IRLAN_ARRAY, 0, 0, string,
  786. string_len);
  787. }
  788. /*
  789. * Function insert_array_param(skb, param, value, len_value)
  790. *
  791. * Insert array parameter into frame
  792. *
  793. */
  794. int irlan_insert_array_param(struct sk_buff *skb, char *name, __u8 *array,
  795. __u16 array_len)
  796. {
  797. return __irlan_insert_param(skb, name, IRLAN_ARRAY, 0, 0, array,
  798. array_len);
  799. }
  800. /*
  801. * Function insert_param (skb, param, value, byte)
  802. *
  803. * Insert parameter at end of buffer, structure of a parameter is:
  804. *
  805. * -----------------------------------------------------------------------
  806. * | Name Length[1] | Param Name[1..255] | Val Length[2] | Value[0..1016]|
  807. * -----------------------------------------------------------------------
  808. */
  809. static int __irlan_insert_param(struct sk_buff *skb, char *param, int type,
  810. __u8 value_byte, __u16 value_short,
  811. __u8 *value_array, __u16 value_len)
  812. {
  813. __u8 *frame;
  814. __u8 param_len;
  815. __u16 tmp_le; /* Temporary value in little endian format */
  816. int n=0;
  817. if (skb == NULL) {
  818. IRDA_DEBUG(2, "%s(), Got NULL skb\n", __FUNCTION__ );
  819. return 0;
  820. }
  821. param_len = strlen(param);
  822. switch (type) {
  823. case IRLAN_BYTE:
  824. value_len = 1;
  825. break;
  826. case IRLAN_SHORT:
  827. value_len = 2;
  828. break;
  829. case IRLAN_ARRAY:
  830. IRDA_ASSERT(value_array != NULL, return 0;);
  831. IRDA_ASSERT(value_len > 0, return 0;);
  832. break;
  833. default:
  834. IRDA_DEBUG(2, "%s(), Unknown parameter type!\n", __FUNCTION__ );
  835. return 0;
  836. break;
  837. }
  838. /* Insert at end of sk-buffer */
  839. frame = skb->tail;
  840. /* Make space for data */
  841. if (skb_tailroom(skb) < (param_len+value_len+3)) {
  842. IRDA_DEBUG(2, "%s(), No more space at end of skb\n", __FUNCTION__ );
  843. return 0;
  844. }
  845. skb_put(skb, param_len+value_len+3);
  846. /* Insert parameter length */
  847. frame[n++] = param_len;
  848. /* Insert parameter */
  849. memcpy(frame+n, param, param_len); n += param_len;
  850. /* Insert value length (2 byte little endian format, LSB first) */
  851. tmp_le = cpu_to_le16(value_len);
  852. memcpy(frame+n, &tmp_le, 2); n += 2; /* To avoid alignment problems */
  853. /* Insert value */
  854. switch (type) {
  855. case IRLAN_BYTE:
  856. frame[n++] = value_byte;
  857. break;
  858. case IRLAN_SHORT:
  859. tmp_le = cpu_to_le16(value_short);
  860. memcpy(frame+n, &tmp_le, 2); n += 2;
  861. break;
  862. case IRLAN_ARRAY:
  863. memcpy(frame+n, value_array, value_len); n+=value_len;
  864. break;
  865. default:
  866. break;
  867. }
  868. IRDA_ASSERT(n == (param_len+value_len+3), return 0;);
  869. return param_len+value_len+3;
  870. }
  871. /*
  872. * Function irlan_extract_param (buf, name, value, len)
  873. *
  874. * Extracts a single parameter name/value pair from buffer and updates
  875. * the buffer pointer to point to the next name/value pair.
  876. */
  877. int irlan_extract_param(__u8 *buf, char *name, char *value, __u16 *len)
  878. {
  879. __u8 name_len;
  880. __u16 val_len;
  881. int n=0;
  882. IRDA_DEBUG(4, "%s()\n", __FUNCTION__ );
  883. /* get length of parameter name (1 byte) */
  884. name_len = buf[n++];
  885. if (name_len > 254) {
  886. IRDA_DEBUG(2, "%s(), name_len > 254\n", __FUNCTION__ );
  887. return -RSP_INVALID_COMMAND_FORMAT;
  888. }
  889. /* get parameter name */
  890. memcpy(name, buf+n, name_len);
  891. name[name_len] = '\0';
  892. n+=name_len;
  893. /*
  894. * Get length of parameter value (2 bytes in little endian
  895. * format)
  896. */
  897. memcpy(&val_len, buf+n, 2); /* To avoid alignment problems */
  898. le16_to_cpus(&val_len); n+=2;
  899. if (val_len > 1016) {
  900. IRDA_DEBUG(2, "%s(), parameter length to long\n", __FUNCTION__ );
  901. return -RSP_INVALID_COMMAND_FORMAT;
  902. }
  903. *len = val_len;
  904. /* get parameter value */
  905. memcpy(value, buf+n, val_len);
  906. value[val_len] = '\0';
  907. n+=val_len;
  908. IRDA_DEBUG(4, "Parameter: %s ", name);
  909. IRDA_DEBUG(4, "Value: %s\n", value);
  910. return n;
  911. }
  912. #ifdef CONFIG_PROC_FS
  913. /*
  914. * Start of reading /proc entries.
  915. * Return entry at pos,
  916. * or start_token to indicate print header line
  917. * or NULL if end of file
  918. */
  919. static void *irlan_seq_start(struct seq_file *seq, loff_t *pos)
  920. {
  921. int i = 1;
  922. struct irlan_cb *self;
  923. rcu_read_lock();
  924. if (*pos == 0)
  925. return SEQ_START_TOKEN;
  926. list_for_each_entry(self, &irlans, dev_list) {
  927. if (*pos == i)
  928. return self;
  929. ++i;
  930. }
  931. return NULL;
  932. }
  933. /* Return entry after v, and increment pos */
  934. static void *irlan_seq_next(struct seq_file *seq, void *v, loff_t *pos)
  935. {
  936. struct list_head *nxt;
  937. ++*pos;
  938. if (v == SEQ_START_TOKEN)
  939. nxt = irlans.next;
  940. else
  941. nxt = ((struct irlan_cb *)v)->dev_list.next;
  942. return (nxt == &irlans) ? NULL
  943. : list_entry(nxt, struct irlan_cb, dev_list);
  944. }
  945. /* End of reading /proc file */
  946. static void irlan_seq_stop(struct seq_file *seq, void *v)
  947. {
  948. rcu_read_unlock();
  949. }
  950. /*
  951. * Show one entry in /proc file.
  952. */
  953. static int irlan_seq_show(struct seq_file *seq, void *v)
  954. {
  955. if (v == SEQ_START_TOKEN)
  956. seq_puts(seq, "IrLAN instances:\n");
  957. else {
  958. struct irlan_cb *self = v;
  959. IRDA_ASSERT(self != NULL, return -1;);
  960. IRDA_ASSERT(self->magic == IRLAN_MAGIC, return -1;);
  961. seq_printf(seq,"ifname: %s,\n",
  962. self->dev->name);
  963. seq_printf(seq,"client state: %s, ",
  964. irlan_state[ self->client.state]);
  965. seq_printf(seq,"provider state: %s,\n",
  966. irlan_state[ self->provider.state]);
  967. seq_printf(seq,"saddr: %#08x, ",
  968. self->saddr);
  969. seq_printf(seq,"daddr: %#08x\n",
  970. self->daddr);
  971. seq_printf(seq,"version: %d.%d,\n",
  972. self->version[1], self->version[0]);
  973. seq_printf(seq,"access type: %s\n",
  974. irlan_access[self->client.access_type]);
  975. seq_printf(seq,"media: %s\n",
  976. irlan_media[self->media]);
  977. seq_printf(seq,"local filter:\n");
  978. seq_printf(seq,"remote filter: ");
  979. irlan_print_filter(seq, self->client.filter_type);
  980. seq_printf(seq,"tx busy: %s\n",
  981. netif_queue_stopped(self->dev) ? "TRUE" : "FALSE");
  982. seq_putc(seq,'\n');
  983. }
  984. return 0;
  985. }
  986. static struct seq_operations irlan_seq_ops = {
  987. .start = irlan_seq_start,
  988. .next = irlan_seq_next,
  989. .stop = irlan_seq_stop,
  990. .show = irlan_seq_show,
  991. };
  992. static int irlan_seq_open(struct inode *inode, struct file *file)
  993. {
  994. return seq_open(file, &irlan_seq_ops);
  995. }
  996. #endif
  997. MODULE_AUTHOR("Dag Brattli <dagb@cs.uit.no>");
  998. MODULE_DESCRIPTION("The Linux IrDA LAN protocol");
  999. MODULE_LICENSE("GPL");
  1000. module_param(eth, bool, 0);
  1001. MODULE_PARM_DESC(eth, "Name devices ethX (0) or irlanX (1)");
  1002. module_param(access, int, 0);
  1003. MODULE_PARM_DESC(access, "Access type DIRECT=1, PEER=2, HOSTED=3");
  1004. module_init(irlan_init);
  1005. module_exit(irlan_cleanup);