summaryrefslogtreecommitdiffstats
path: root/net/irda/irlmp.c
diff options
context:
space:
mode:
authorRalf Baechle <ralf@linux-mips.org>1999-06-13 16:29:25 +0000
committerRalf Baechle <ralf@linux-mips.org>1999-06-13 16:29:25 +0000
commitdb7d4daea91e105e3859cf461d7e53b9b77454b2 (patch)
tree9bb65b95440af09e8aca63abe56970dd3360cc57 /net/irda/irlmp.c
parent9c1c01ead627bdda9211c9abd5b758d6c687d8ac (diff)
Merge with Linux 2.2.8.
Diffstat (limited to 'net/irda/irlmp.c')
-rw-r--r--net/irda/irlmp.c1094
1 files changed, 627 insertions, 467 deletions
diff --git a/net/irda/irlmp.c b/net/irda/irlmp.c
index c4f7c2b8d..d76661b6c 100644
--- a/net/irda/irlmp.c
+++ b/net/irda/irlmp.c
@@ -1,12 +1,12 @@
/*********************************************************************
*
* Filename: irlmp.c
- * Version: 0.8
+ * Version: 0.9
* Description: IrDA Link Management Protocol (LMP) layer
- * Status: Experimental.
+ * Status: Stable.
* Author: Dag Brattli <dagb@cs.uit.no>
* Created at: Sun Aug 17 20:54:32 1997
- * Modified at: Sat Jan 16 22:13:20 1999
+ * Modified at: Fri Apr 23 09:13:24 1999
* Modified by: Dag Brattli <dagb@cs.uit.no>
*
* Copyright (c) 1998 Dag Brattli <dagb@cs.uit.no>,
@@ -30,6 +30,9 @@
#include <linux/types.h>
#include <linux/proc_fs.h>
#include <linux/init.h>
+#include <linux/kmod.h>
+#include <linux/random.h>
+#include <linux/irda.h>
#include <net/irda/irda.h>
#include <net/irda/irmod.h>
@@ -39,18 +42,28 @@
#include <net/irda/iriap.h>
#include <net/irda/irlmp.h>
#include <net/irda/irlmp_frame.h>
-#include <linux/kmod.h>
/* Master structure */
struct irlmp_cb *irlmp = NULL;
-int sysctl_discovery = 0;
+/* These can be altered by the sysctl interface */
+int sysctl_discovery = 0;
+int sysctl_discovery_slots = 6;
char sysctl_devname[65];
+char *lmp_reasons[] = {
+ "ERROR, NOT USED",
+ "LM_USER_REQUEST",
+ "LM_LAP_DISCONNECT",
+ "LM_CONNECT_FAILURE",
+ "LM_LAP_RESET",
+ "LM_INIT_DISCONNECT",
+ "ERROR, NOT USED",
+};
+
__u8 *irlmp_hint_to_service( __u8 *hint);
#ifdef CONFIG_PROC_FS
-int irlmp_proc_read( char *buf, char **start, off_t offset, int len,
- int unused);
+int irlmp_proc_read(char *buf, char **start, off_t offst, int len, int unused);
#endif
/*
@@ -61,8 +74,6 @@ int irlmp_proc_read( char *buf, char **start, off_t offset, int len,
*/
__initfunc(int irlmp_init(void))
{
- DEBUG( 4, "--> irlmp_init\n");
-
/* Initialize the irlmp structure. */
if ( irlmp == NULL) {
irlmp = kmalloc( sizeof(struct irlmp_cb), GFP_KERNEL);
@@ -73,19 +84,21 @@ __initfunc(int irlmp_init(void))
irlmp->magic = LMP_MAGIC;
- irlmp->registry = hashbin_new( HB_LOCAL);
- irlmp->links = hashbin_new( HB_LOCAL);
- irlmp->unconnected_lsaps = hashbin_new( HB_GLOBAL);
+ irlmp->clients = hashbin_new(HB_GLOBAL);
+ irlmp->services = hashbin_new(HB_GLOBAL);
+ irlmp->links = hashbin_new(HB_GLOBAL);
+ irlmp->unconnected_lsaps = hashbin_new(HB_GLOBAL);
+ irlmp->cachelog = hashbin_new(HB_GLOBAL);
irlmp->free_lsap_sel = 0x10; /* Servers use 0x00-0x0f */
#ifdef CONFIG_IRDA_CACHE_LAST_LSAP
irlmp->cache.valid = FALSE;
#endif
- strcpy( sysctl_devname, "Linux");
+ strcpy(sysctl_devname, "Linux");
/* Do discovery every 3 seconds */
- init_timer( &irlmp->discovery_timer);
- irlmp_start_discovery_timer( irlmp, 600);
+ init_timer(&irlmp->discovery_timer);
+ irlmp_start_discovery_timer(irlmp, 600);
return 0;
}
@@ -99,18 +112,19 @@ __initfunc(int irlmp_init(void))
void irlmp_cleanup(void)
{
/* Check for main structure */
- ASSERT( irlmp != NULL, return;);
- ASSERT( irlmp->magic == LMP_MAGIC, return;);
+ ASSERT(irlmp != NULL, return;);
+ ASSERT(irlmp->magic == LMP_MAGIC, return;);
- del_timer( &irlmp->discovery_timer);
+ del_timer(&irlmp->discovery_timer);
- /* FIXME, we need a special function to deallocate LAPs */
- hashbin_delete( irlmp->links, (FREE_FUNC) kfree);
- hashbin_delete( irlmp->unconnected_lsaps, (FREE_FUNC) kfree);
- hashbin_delete( irlmp->registry, (FREE_FUNC) kfree);
+ hashbin_delete(irlmp->links, (FREE_FUNC) kfree);
+ hashbin_delete(irlmp->unconnected_lsaps, (FREE_FUNC) kfree);
+ hashbin_delete(irlmp->clients, (FREE_FUNC) kfree);
+ hashbin_delete(irlmp->services, (FREE_FUNC) kfree);
+ hashbin_delete(irlmp->cachelog, (FREE_FUNC) kfree);
/* De-allocate main structure */
- kfree( irlmp);
+ kfree(irlmp);
irlmp = NULL;
}
@@ -120,20 +134,20 @@ void irlmp_cleanup(void)
* Register with IrLMP and create a local LSAP,
* returns handle to LSAP.
*/
-struct lsap_cb *irlmp_open_lsap( __u8 slsap_sel, struct notify_t *notify)
+struct lsap_cb *irlmp_open_lsap(__u8 slsap_sel, struct notify_t *notify)
{
struct lsap_cb *self;
- ASSERT( notify != NULL, return NULL;);
- ASSERT( irlmp != NULL, return NULL;);
- ASSERT( irlmp->magic == LMP_MAGIC, return NULL;);
+ ASSERT(notify != NULL, return NULL;);
+ ASSERT(irlmp != NULL, return NULL;);
+ ASSERT(irlmp->magic == LMP_MAGIC, return NULL;);
- DEBUG( 4, "irlmp_open_lsap(), slsap_sel=%02x\n", slsap_sel);
+ DEBUG(4, __FUNCTION__ "(), slsap_sel=%02x\n", slsap_sel);
/*
* Does the client care which Source LSAP selector it gets?
*/
- if ( slsap_sel == LSAP_ANY) {
+ if (slsap_sel == LSAP_ANY) {
/*
* Find unused LSAP
*/
@@ -145,40 +159,39 @@ struct lsap_cb *irlmp_open_lsap( __u8 slsap_sel, struct notify_t *notify)
* Client wants specific LSAP, so check if it's already
* in use
*/
- if ( irlmp_slsap_inuse( slsap_sel)) {
+ if (irlmp_slsap_inuse(slsap_sel)) {
return NULL;
}
- if ( slsap_sel > irlmp->free_lsap_sel)
- irlmp->free_lsap_sel = slsap_sel+1;
}
/*
* Allocate new instance of a LSAP connection
*/
- self = kmalloc( sizeof(struct lsap_cb), GFP_ATOMIC);
- if ( self == NULL) {
+ self = kmalloc(sizeof(struct lsap_cb), GFP_ATOMIC);
+ if (self == NULL) {
printk( KERN_ERR "IrLMP: Can't allocate memory for "
"LSAP control block!\n");
return NULL;
}
- memset( self, 0, sizeof(struct lsap_cb));
+ memset(self, 0, sizeof(struct lsap_cb));
self->magic = LMP_LSAP_MAGIC;
self->slsap_sel = slsap_sel;
self->dlsap_sel = LSAP_ANY;
+ self->connected = FALSE;
- init_timer( &self->watchdog_timer);
+ init_timer(&self->watchdog_timer);
- ASSERT( notify->instance != NULL, return NULL;);
+ ASSERT(notify->instance != NULL, return NULL;);
self->notify = *notify;
- irlmp_next_lsap_state( self, LSAP_DISCONNECTED);
+ irlmp_next_lsap_state(self, LSAP_DISCONNECTED);
/*
* Insert into queue of unconnected LSAPs
*/
- hashbin_insert( irlmp->unconnected_lsaps, (QUEUE *) self,
- self->slsap_sel, NULL);
+ hashbin_insert(irlmp->unconnected_lsaps, (QUEUE *) self, (int) self,
+ NULL);
return self;
}
@@ -186,70 +199,62 @@ struct lsap_cb *irlmp_open_lsap( __u8 slsap_sel, struct notify_t *notify)
/*
* Function irlmp_close_lsap (self)
*
- * Remove an instance of a LSAP
+ * Remove an instance of LSAP
*/
-static void __irlmp_close_lsap( struct lsap_cb *self)
+static void __irlmp_close_lsap(struct lsap_cb *self)
{
- DEBUG( 4, "irlmp_close()\n");
+ DEBUG(4, __FUNCTION__ "()\n");
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
+ ASSERT(self != NULL, return;);
+ ASSERT(self->magic == LMP_LSAP_MAGIC, return;);
/*
* Set some of the variables to preset values
*/
self->magic = ~LMP_LSAP_MAGIC;
- del_timer( &self->watchdog_timer); /* Important! */
+ del_timer(&self->watchdog_timer); /* Important! */
#ifdef CONFIG_IRDA_CACHE_LAST_LSAP
- ASSERT( irlmp != NULL, return;);
+ ASSERT(irlmp != NULL, return;);
irlmp->cache.valid = FALSE;
#endif
- /*
- * Deallocate structure
- */
- kfree( self);
-
- DEBUG( 4, "irlmp_close() -->\n");
+ kfree(self);
}
/*
* Function irlmp_close_lsap (self)
*
- *
+ * Close and remove LSAP
*
*/
-void irlmp_close_lsap( struct lsap_cb *self)
+void irlmp_close_lsap(struct lsap_cb *self)
{
struct lap_cb *lap;
- struct lsap_cb *lsap;
+ struct lsap_cb *lsap = NULL;
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
-
- lap = self->lap;
+ ASSERT(self != NULL, return;);
+ ASSERT(self->magic == LMP_LSAP_MAGIC, return;);
/*
* Find out if we should remove this LSAP from a link or from the
* list of unconnected lsaps (not associated with a link)
*/
- if ( lap == NULL) {
- lsap = hashbin_remove( irlmp->unconnected_lsaps,
- self->slsap_sel, NULL);
- } else {
- ASSERT( lap != NULL, return;);
- ASSERT( lap->magic == LMP_LAP_MAGIC, return;);
-
- lsap = hashbin_remove( lap->lsaps, self->slsap_sel, NULL);
+ lap = self->lap;
+ if (lap) {
+ ASSERT(lap->magic == LMP_LAP_MAGIC, return;);
+ lsap = hashbin_remove(lap->lsaps, (int) self, NULL);
+ }
+ /* Check if we found the LSAP! If not then try the unconnected lsaps */
+ if (!lsap) {
+ lsap = hashbin_remove(irlmp->unconnected_lsaps, (int) self,
+ NULL);
}
- if ( lsap == NULL) {
- DEBUG( 0, __FUNCTION__
+ if (!lsap) {
+ DEBUG(0, __FUNCTION__
"(), Looks like somebody has removed me already!\n");
return;
}
- ASSERT( lsap == self, return;);
-
- __irlmp_close_lsap( self);
+ __irlmp_close_lsap(self);
}
/*
@@ -259,47 +264,47 @@ void irlmp_close_lsap( struct lsap_cb *self)
* instances of the IrLAP layer, each connected to different IrDA ports
*
*/
-void irlmp_register_irlap( struct irlap_cb *irlap, __u32 saddr,
- struct notify_t *notify)
+void irlmp_register_link(struct irlap_cb *irlap, __u32 saddr,
+ struct notify_t *notify)
{
struct lap_cb *lap;
- DEBUG( 4, __FUNCTION__ "(), Registered IrLAP, saddr = %08x\n",
- saddr);
+ DEBUG(4, __FUNCTION__ "(), Registered IrLAP, saddr = %08x\n", saddr);
- ASSERT( irlmp != NULL, return;);
- ASSERT( irlmp->magic == LMP_MAGIC, return;);
- ASSERT( notify != NULL, return;);
+ ASSERT(irlmp != NULL, return;);
+ ASSERT(irlmp->magic == LMP_MAGIC, return;);
+ ASSERT(notify != NULL, return;);
/*
* Allocate new instance of a LSAP connection
*/
- lap = kmalloc( sizeof(struct lap_cb), GFP_KERNEL);
- if ( lap == NULL) {
- printk( KERN_ERR "IrLMP: Can't allocate memory for "
- "LAP control block!\n");
+ lap = kmalloc(sizeof(struct lap_cb), GFP_KERNEL);
+ if (lap == NULL) {
+ DEBUG(3, __FUNCTION__ "(), unable to kmalloc\n");
return;
}
- memset( lap, 0, sizeof(struct lap_cb));
+ memset(lap, 0, sizeof(struct lap_cb));
lap->irlap = irlap;
lap->magic = LMP_LAP_MAGIC;
lap->saddr = saddr;
- lap->lsaps = hashbin_new( HB_GLOBAL);
- lap->cachelog = hashbin_new( HB_LOCAL);
+ lap->daddr = DEV_ADDR_ANY;
+ lap->lsaps = hashbin_new(HB_GLOBAL);
- irlmp_next_lap_state( lap, LAP_STANDBY);
+ irlmp_next_lap_state(lap, LAP_STANDBY);
+ init_timer(&lap->idle_timer);
+
/*
* Insert into queue of unconnected LSAPs
*/
- hashbin_insert( irlmp->links, (QUEUE *) lap, lap->saddr, NULL);
+ hashbin_insert(irlmp->links, (QUEUE *) lap, lap->saddr, NULL);
/*
* We set only this variable so IrLAP can tell us on which link the
* different events happened on
*/
- irda_notify_init( notify);
+ irda_notify_init(notify);
notify->instance = lap;
}
@@ -309,36 +314,23 @@ void irlmp_register_irlap( struct irlap_cb *irlap, __u32 saddr,
* IrLAP layer has been removed!
*
*/
-void irlmp_unregister_irlap( __u32 saddr)
+void irlmp_unregister_link(__u32 saddr)
{
- struct lap_cb *self;
+ struct lap_cb *link;
- DEBUG( 4, __FUNCTION__ "()\n");
+ DEBUG(4, __FUNCTION__ "()\n");
- self = hashbin_remove( irlmp->links, saddr, NULL);
- if ( self != NULL) {
- ASSERT( self->magic == LMP_LAP_MAGIC, return;);
+ link = hashbin_remove(irlmp->links, saddr, NULL);
+ if (link) {
+ ASSERT(link->magic == LMP_LAP_MAGIC, return;);
- self->magic = ~LMP_LAP_MAGIC;
- kfree( self);
- } else {
- DEBUG( 0, "irlmp_unregister_irlap(), Didn't find LAP!\n");
- }
-}
-
-void dump_discoveries( hashbin_t *log)
-{
- DISCOVERY *d;
+ /* Remove all discoveries discovered at this link */
+ irlmp_expire_discoveries(irlmp->cachelog, link->saddr, TRUE);
- ASSERT( log != NULL, return;);
+ del_timer(&link->idle_timer);
- d = (DISCOVERY *) hashbin_get_first( log);
- while( d != NULL) {
- DEBUG( 0, "Discovery:\n");
- DEBUG( 0, " daddr=%08x\n", d->daddr);
- DEBUG( 0, " name=%s\n", d->info);
-
- d = (DISCOVERY *) hashbin_get_next( log);
+ link->magic = 0;
+ kfree(link);
}
}
@@ -348,100 +340,98 @@ void dump_discoveries( hashbin_t *log)
* Connect with a peer LSAP
*
*/
-void irlmp_connect_request( struct lsap_cb *self, __u8 dlsap_sel, __u32 daddr,
- struct qos_info *qos, struct sk_buff *userdata)
+int irlmp_connect_request(struct lsap_cb *self, __u8 dlsap_sel,
+ __u32 saddr, __u32 daddr,
+ struct qos_info *qos, struct sk_buff *userdata)
{
struct sk_buff *skb = NULL;
struct lap_cb *lap;
struct lsap_cb *lsap;
- unsigned long flags;
+ discovery_t *discovery;
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
+ ASSERT( self != NULL, return -1;);
+ ASSERT( self->magic == LMP_LSAP_MAGIC, return -1;);
- DEBUG( 4, "irlmp_connect_request(), "
- "slsap_sel=%02x, dlsap_sel=%02x, daddr=%08x\n",
- self->slsap_sel, dlsap_sel, daddr);
+ DEBUG(2, __FUNCTION__
+ "(), slsap_sel=%02x, dlsap_sel=%02x, saddr=%08x, daddr=%08x\n",
+ self->slsap_sel, dlsap_sel, saddr, daddr);
+
+ if ( self->connected)
+ return -EISCONN;
- if ( self->connected) {
- DEBUG( 0, __FUNCTION__ "(), Error: already connected!!\n");
-
- return;
- }
+ /* Client must supply destination device address */
+ if (!daddr)
+ return -EINVAL;
/* Any userdata? */
- if ( userdata == NULL) {
- skb = dev_alloc_skb( 64);
- if (skb == NULL) {
- DEBUG( 0, __FUNCTION__
- "(), Could not allocate sk_buff of length %d\n",
- 64);
- return;
- }
- skb_reserve( skb, LMP_CONTROL_HEADER+LAP_HEADER);
+ if (userdata == NULL) {
+ skb = dev_alloc_skb(64);
+ if (!skb)
+ return -ENOMEM;
+
+ skb_reserve(skb, LMP_CONTROL_HEADER+LAP_HEADER);
} else
skb = userdata;
/* Make room for MUX control header ( 3 bytes) */
- ASSERT( skb_headroom( skb) >= LMP_CONTROL_HEADER, return;);
- skb_push( skb, LMP_CONTROL_HEADER);
+ ASSERT(skb_headroom(skb) >= LMP_CONTROL_HEADER, return -1;);
+ skb_push(skb, LMP_CONTROL_HEADER);
self->dlsap_sel = dlsap_sel;
self->tmp_skb = skb;
- /*
- * Find out which link to connect on, and make sure nothing strange
- * happens while we traverse the list
+ /*
+ * Find the link to where we should try to connect since there may
+ * be more than one IrDA port on this machine. If the client has
+ * passed us the saddr (and already knows which link to use), then
+ * we use that to find the link, if not then we have to look in the
+ * discovery log and check if any of the links has discovered a
+ * device with the given daddr
*/
- save_flags( flags);
- cli();
-
- lap = (struct lap_cb *) hashbin_get_first( irlmp->links);
- while ( lap != NULL) {
- ASSERT( lap->magic == LMP_LAP_MAGIC, return;);
- /* dump_discoveries( lap->cachelog); */
+ if (!saddr) {
+ discovery = hashbin_find(irlmp->cachelog, daddr, NULL);
+ if (discovery)
+ saddr = discovery->saddr;
+ }
+ lap = hashbin_find(irlmp->links, saddr, NULL);
+ if (lap == NULL) {
+ DEBUG(1, __FUNCTION__ "(), Unable to find a usable link!\n");
+ return -EHOSTUNREACH;
+ }
- if ( hashbin_find( lap->cachelog, daddr, NULL)) {
- DEBUG( 4, "irlmp_connect_request() found link to connect on!\n");
- self->lap = lap;
- break;
- }
- lap = (struct lap_cb *) hashbin_get_next( irlmp->links);
+ if (lap->daddr == DEV_ADDR_ANY)
+ lap->daddr = daddr;
+ else if (lap->daddr != daddr) {
+ DEBUG(0, __FUNCTION__ "(), sorry, but link is busy!\n");
+ return -EBUSY;
}
- restore_flags(flags);
-
+
+ self->lap = lap;
+
/*
* Remove LSAP from list of unconnected LSAPs and insert it into the
- * list of connected LSAPs for the particular link */
- lsap = hashbin_remove( irlmp->unconnected_lsaps, self->slsap_sel,
- NULL);
-
- /* Check if we found a link to connect on */
- if ( self->lap == NULL) {
- DEBUG( 0, __FUNCTION__ "(), Unable to find a usable link!\n");
- return;
- }
+ * list of connected LSAPs for the particular link
+ */
+ lsap = hashbin_remove(irlmp->unconnected_lsaps, (int) self, NULL);
- ASSERT( lsap != NULL, return;);
- ASSERT( lsap->magic == LMP_LSAP_MAGIC, return;);
- ASSERT( lsap->lap != NULL, return;);
- ASSERT( lsap->lap->magic == LMP_LAP_MAGIC, return;);
+ ASSERT(lsap != NULL, return -1;);
+ ASSERT(lsap->magic == LMP_LSAP_MAGIC, return -1;);
+ ASSERT(lsap->lap != NULL, return -1;);
+ ASSERT(lsap->lap->magic == LMP_LAP_MAGIC, return -1;);
- hashbin_insert( self->lap->lsaps, (QUEUE *) self, self->slsap_sel,
- NULL);
+ hashbin_insert(self->lap->lsaps, (QUEUE *) self, (int) self, NULL);
self->connected = TRUE;
-
+
/*
* User supplied qos specifications?
*/
- if ( qos)
+ if (qos)
self->qos = *qos;
- DEBUG( 4, "*** Connecting SLSAP=%02x, DLSAP= %02x\n",
- self->slsap_sel, self->dlsap_sel);
-
- irlmp_do_lsap_event( self, LM_CONNECT_REQUEST, skb);
+ irlmp_do_lsap_event(self, LM_CONNECT_REQUEST, skb);
+
+ return 0;
}
/*
@@ -450,29 +440,28 @@ void irlmp_connect_request( struct lsap_cb *self, __u8 dlsap_sel, __u32 daddr,
* Incomming connection
*
*/
-void irlmp_connect_indication( struct lsap_cb *self, struct sk_buff *skb)
+void irlmp_connect_indication(struct lsap_cb *self, struct sk_buff *skb)
{
int max_seg_size;
- DEBUG( 4, "irlmp_connect_indication()\n");
+ DEBUG(3, __FUNCTION__ "()\n");
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
- ASSERT( skb != NULL, return;);
- ASSERT( self->lap != NULL, return;);
+ ASSERT(self != NULL, return;);
+ ASSERT(self->magic == LMP_LSAP_MAGIC, return;);
+ ASSERT(skb != NULL, return;);
+ ASSERT(self->lap != NULL, return;);
self->qos = *self->lap->qos;
max_seg_size = self->lap->qos->data_size.value;
- DEBUG( 4, __FUNCTION__ "(), max_seg_size=%d\n", max_seg_size);
+ DEBUG(4, __FUNCTION__ "(), max_seg_size=%d\n", max_seg_size);
/* Hide LMP_CONTROL_HEADER header from layer above */
- skb_pull( skb, LMP_CONTROL_HEADER);
+ skb_pull(skb, LMP_CONTROL_HEADER);
- if ( self->notify.connect_indication)
- self->notify.connect_indication( self->notify.instance, self,
- &self->qos, max_seg_size,
- skb);
+ if (self->notify.connect_indication)
+ self->notify.connect_indication(self->notify.instance, self,
+ &self->qos, max_seg_size, skb);
}
/*
@@ -483,7 +472,7 @@ void irlmp_connect_indication( struct lsap_cb *self, struct sk_buff *skb)
*/
void irlmp_connect_response( struct lsap_cb *self, struct sk_buff *userdata)
{
- DEBUG( 4, "irlmp_connect_response()\n");
+ DEBUG(3, __FUNCTION__ "()\n");
ASSERT( self != NULL, return;);
ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
@@ -506,11 +495,11 @@ void irlmp_connect_response( struct lsap_cb *self, struct sk_buff *userdata)
*
* LSAP connection confirmed peer device!
*/
-void irlmp_connect_confirm( struct lsap_cb *self, struct sk_buff *skb)
+void irlmp_connect_confirm(struct lsap_cb *self, struct sk_buff *skb)
{
int max_seg_size;
- DEBUG( 4, __FUNCTION__ "()\n");
+ DEBUG(3, __FUNCTION__ "()\n");
ASSERT( skb != NULL, return;);
ASSERT( self != NULL, return;);
@@ -532,23 +521,63 @@ void irlmp_connect_confirm( struct lsap_cb *self, struct sk_buff *skb)
}
/*
+ * Function irlmp_dup (orig, instance)
+ *
+ * Duplicate LSAP, can be used by servers to confirm a connection on a
+ * new LSAP so it can keep listening on the old one.
+ *
+ */
+struct lsap_cb *irlmp_dup(struct lsap_cb *orig, void *instance)
+{
+ struct lsap_cb *new;
+
+ DEBUG(1, __FUNCTION__ "()\n");
+
+ /* Only allowed to duplicate unconnected LSAP's */
+ if (!hashbin_find(irlmp->unconnected_lsaps, (int) orig, NULL)) {
+ DEBUG(0, __FUNCTION__ "(), unable to find LSAP\n");
+ return NULL;
+ }
+ new = kmalloc(sizeof(struct lsap_cb), GFP_ATOMIC);
+ if (!new) {
+ DEBUG(0, __FUNCTION__ "(), unable to kmalloc\n");
+ return NULL;
+ }
+ /* Dup */
+ memcpy(new, orig, sizeof(struct lsap_cb));
+ new->notify.instance = instance;
+
+ init_timer(&new->watchdog_timer);
+
+ hashbin_insert(irlmp->unconnected_lsaps, (QUEUE *) new, (int) new,
+ NULL);
+
+ /* Make sure that we invalidate the cache */
+#ifdef CONFIG_IRDA_CACHE_LAST_LSAP
+ irlmp->cache.valid = FALSE;
+#endif /* CONFIG_IRDA_CACHE_LAST_LSAP */
+
+ return new;
+}
+
+/*
* Function irlmp_disconnect_request (handle, userdata)
*
* The service user is requesting disconnection, this will not remove the
* LSAP, but only mark it as disconnected
*/
-void irlmp_disconnect_request( struct lsap_cb *self, struct sk_buff *userdata)
+void irlmp_disconnect_request(struct lsap_cb *self, struct sk_buff *userdata)
{
struct lsap_cb *lsap;
- DEBUG( 4, "irlmp_disconnect_request()\n");
+ DEBUG( 4, __FUNCTION__ "()\n");
ASSERT( self != NULL, return;);
ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
/* Already disconnected? */
if ( !self->connected) {
- DEBUG( 0, __FUNCTION__ "(), already disconnected!\n");
+ DEBUG( 1, __FUNCTION__ "(), already disconnected!\n");
return;
}
@@ -571,14 +600,14 @@ void irlmp_disconnect_request( struct lsap_cb *self, struct sk_buff *userdata)
ASSERT( self->lap->magic == LMP_LAP_MAGIC, return;);
ASSERT( self->lap->lsaps != NULL, return;);
- lsap = hashbin_remove( self->lap->lsaps, self->slsap_sel, NULL);
+ lsap = hashbin_remove(self->lap->lsaps, (int) self, NULL);
ASSERT( lsap != NULL, return;);
ASSERT( lsap->magic == LMP_LSAP_MAGIC, return;);
ASSERT( lsap == self, return;);
- hashbin_insert( irlmp->unconnected_lsaps, (QUEUE *) self,
- self->slsap_sel, NULL);
+ hashbin_insert(irlmp->unconnected_lsaps, (QUEUE *) self, (int) self,
+ NULL);
/* Reset some values */
self->connected = FALSE;
@@ -596,37 +625,32 @@ void irlmp_disconnect_indication( struct lsap_cb *self, LM_REASON reason,
{
struct lsap_cb *lsap;
- DEBUG( 4, __FUNCTION__ "()\n");
+ DEBUG( 1, __FUNCTION__ "(), reason=%s\n", lmp_reasons[reason]);
ASSERT( self != NULL, return;);
ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
ASSERT( self->connected == TRUE, return;);
- DEBUG( 4, __FUNCTION__ "(), slsap_sel=%02x, dlsap_sel=%02x\n",
+ DEBUG( 3, __FUNCTION__ "(), slsap_sel=%02x, dlsap_sel=%02x\n",
self->slsap_sel, self->dlsap_sel);
self->connected = FALSE;
self->dlsap_sel = LSAP_ANY;
/*
- * Remove assosiation betwen this LSAP and the kink it used
+ * Remove association between this LSAP and the link it used
*/
ASSERT( self->lap != NULL, return;);
ASSERT( self->lap->lsaps != NULL, return;);
- lsap = hashbin_remove( self->lap->lsaps, self->slsap_sel, NULL);
+ lsap = hashbin_remove( self->lap->lsaps, (int) self, NULL);
ASSERT( lsap != NULL, return;);
ASSERT( lsap == self, return;);
- hashbin_insert( irlmp->unconnected_lsaps, (QUEUE *) lsap,
- lsap->slsap_sel, NULL);
+ hashbin_insert(irlmp->unconnected_lsaps, (QUEUE *) lsap, (int) lsap,
+ NULL);
self->lap = NULL;
-
- /* FIXME: the reasons should be extracted somewhere else? */
- if ( userdata) {
- DEBUG( 4, __FUNCTION__ "(), reason=%02x\n", userdata->data[3]);
- }
/*
* Inform service user
@@ -638,28 +662,24 @@ void irlmp_disconnect_indication( struct lsap_cb *self, LM_REASON reason,
}
/*
- * Function irlmp_discovery_request (nslots)
+ * Function irlmp_do_discovery (nslots)
*
- * Do a discovery of devices in front of the computer
+ * Do some discovery on all links
*
*/
-void irlmp_discovery_request( int nslots)
+void irlmp_do_discovery(int nslots)
{
struct lap_cb *lap;
- DEBUG( 4, __FUNCTION__ "()\n");
-
- ASSERT( irlmp != NULL, return;);
-
- if ( !sysctl_discovery)
- return;
+ /* Make sure value is sane */
+ if ((nslots != 1) && (nslots != 6) && (nslots != 8)&&(nslots != 16)) {
+ printk(KERN_WARNING __FUNCTION__
+ "(), invalid value for number of slots!\n");
+ nslots = sysctl_discovery_slots = 8;
+ }
- /*
- * Construct new discovery info to be used by IrLAP,
- * TODO: no need to do this every time!
- */
- irlmp->discovery_cmd.hint[0] = irlmp->hint[0];
- irlmp->discovery_cmd.hint[1] = irlmp->hint[1];
+ /* Construct new discovery info to be used by IrLAP, */
+ irlmp->discovery_cmd.hints.word = irlmp->hints.word;
/*
* Set character set for device name (we use ASCII), and
@@ -667,138 +687,164 @@ void irlmp_discovery_request( int nslots)
* end
*/
irlmp->discovery_cmd.charset = CS_ASCII;
-
- strncpy( irlmp->discovery_cmd.info, sysctl_devname, 31);
- irlmp->discovery_cmd.info_len = strlen( irlmp->discovery_cmd.info);
+ strncpy(irlmp->discovery_cmd.info, sysctl_devname, 31);
+ irlmp->discovery_cmd.info_len = strlen(irlmp->discovery_cmd.info);
+ irlmp->discovery_cmd.nslots = nslots;
/*
* Try to send discovery packets on all links
*/
- lap = ( struct lap_cb *) hashbin_get_first( irlmp->links);
- while ( lap != NULL) {
- ASSERT( lap->magic == LMP_LAP_MAGIC, return;);
-
- DEBUG( 4, "irlmp_discovery_request() sending request!\n");
- irlmp_do_lap_event( lap, LM_LAP_DISCOVERY_REQUEST, NULL);
+ lap = (struct lap_cb *) hashbin_get_first(irlmp->links);
+ while (lap != NULL) {
+ ASSERT(lap->magic == LMP_LAP_MAGIC, return;);
- lap = ( struct lap_cb *) hashbin_get_next( irlmp->links);
+ if (lap->lap_state == LAP_STANDBY) {
+ /* Expire discoveries discovered on this link */
+ irlmp_expire_discoveries(irlmp->cachelog, lap->saddr,
+ FALSE);
+
+ /* Try to discover */
+ irlmp_do_lap_event(lap, LM_LAP_DISCOVERY_REQUEST,
+ NULL);
+ }
+ lap = (struct lap_cb *) hashbin_get_next(irlmp->links);
}
}
/*
+ * Function irlmp_discovery_request (nslots)
+ *
+ * Do a discovery of devices in front of the computer
+ *
+ */
+void irlmp_discovery_request(int nslots)
+{
+ DEBUG(4, __FUNCTION__ "(), nslots=%d\n", nslots);
+
+ /* Check if user wants to override the default */
+ if (nslots == DISCOVERY_DEFAULT_SLOTS)
+ nslots = sysctl_discovery_slots;
+
+ /*
+ * If discovery is already running, then just return the current
+ * discovery log
+ */
+ if (sysctl_discovery) {
+ DEBUG(2, __FUNCTION__ "() discovery already running, so we"
+ " just return the old discovery log!\n");
+ irlmp_discovery_confirm(irlmp->cachelog);
+ } else
+ irlmp_do_discovery(nslots);
+}
+
+#if 0
+/*
* Function irlmp_check_services (discovery)
*
*
*
*/
-void irlmp_check_services( DISCOVERY *discovery)
+void irlmp_check_services(discovery_t *discovery)
{
- struct irlmp_registration *entry;
+ struct irlmp_client *client;
struct irmanager_event event;
- __u8 *service;
+ __u8 *service_log;
+ __u8 service;
int i = 0;
- printk( KERN_INFO "IrDA Discovered: %s\n", discovery->info);
- printk( KERN_INFO " Services: ");
+ DEBUG(1, "IrDA Discovered: %s\n", discovery->info);
+ DEBUG(1, " Services: ");
- service = irlmp_hint_to_service( discovery->hint);
- if (service != NULL) {
- /*
- * Check all services on the device
- */
- while ( service[i] != S_END) {
- DEBUG( 4, "service=%02x\n", service[i]);
- entry = hashbin_find( irlmp->registry,
- service[i], NULL);
- if ( entry && entry->discovery_callback) {
- DEBUG( 4, "discovery_callback!\n");
- entry->discovery_callback( discovery);
- } else {
- /*
- * Found no clients for dealing with this
- * service, so ask the user space irmanager
- * to try to load the right module for us
- */
-
- event.event = EVENT_DEVICE_DISCOVERED;
- event.service = service[i];
- event.daddr = discovery->daddr;
- sprintf( event.info, "%s",
- discovery->info);
- irmanager_notify( &event);
- }
- i++; /* Next service */
+ service_log = irlmp_hint_to_service(discovery->hints.byte);
+ if (!service_log)
+ return;
+
+ /*
+ * Check all services on the device
+ */
+ while ((service = service_log[i++]) != S_END) {
+ DEBUG( 4, "service=%02x\n", service);
+ client = hashbin_find(irlmp->registry, service, NULL);
+ if (entry && entry->discovery_callback) {
+ DEBUG( 4, "discovery_callback!\n");
+
+ entry->discovery_callback(discovery);
+ } else {
+ /* Don't notify about the ANY service */
+ if (service == S_ANY)
+ continue;
+ /*
+ * Found no clients for dealing with this service,
+ * so ask the user space irmanager to try to load
+ * the right module for us
+ */
+ event.event = EVENT_DEVICE_DISCOVERED;
+ event.service = service;
+ event.daddr = discovery->daddr;
+ sprintf(event.info, "%s", discovery->info);
+ irmanager_notify(&event);
}
- kfree( service);
}
+ kfree(service_log);
}
-
+#endif
/*
- * Function irlmp_discovery_confirm ( self, log)
+ * Function irlmp_notify_client (log)
+ *
+ * Notify all about discovered devices
*
- * Some device(s) answered to our discovery request! Check to see which
- * device it is, and give indication to the client(s)
- *
*/
-void irlmp_discovery_confirm( struct lap_cb *self, hashbin_t *log)
+void irlmp_notify_client(irlmp_client_t *client, hashbin_t *log)
{
- DISCOVERY *discovery;
+ discovery_t *discovery;
+
+ DEBUG(3, __FUNCTION__ "()\n");
- DEBUG( 4, __FUNCTION__ "()\n");
+ /* Check if client wants the whole log */
+ if (client->callback2)
+ client->callback2(log);
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LAP_MAGIC, return;);
-
- /*
- * Now, check all discovered devices (if any)
+ /*
+ * Now, check all discovered devices (if any), and notify client
+ * only about the services that the client is interested in
*/
- discovery = ( DISCOVERY *) hashbin_get_first( log);
- while ( discovery != NULL) {
- self->daddr = discovery->daddr;
-
- DEBUG( 4, "discovery->daddr = 0x%08x\n", discovery->daddr);
+ discovery = (discovery_t *) hashbin_get_first(log);
+ while (discovery != NULL) {
+ DEBUG(3, "discovery->daddr = 0x%08x\n", discovery->daddr);
- irlmp_check_services( discovery);
-
- discovery = ( DISCOVERY *) hashbin_get_next( log);
+ if (client->hint_mask & discovery->hints.word) {
+ if (client->callback1)
+ client->callback1(discovery);
+ }
+ discovery = (discovery_t *) hashbin_get_next(log);
}
}
/*
- * Function irlmp_discovery_indication (discovery)
- *
- * A remote device is discovering us!
+ * Function irlmp_discovery_confirm ( self, log)
*
+ * Some device(s) answered to our discovery request! Check to see which
+ * device it is, and give indication to the client(s)
+ *
*/
-void irlmp_discovery_indication( struct lap_cb *self, DISCOVERY *discovery)
+void irlmp_discovery_confirm(hashbin_t *log)
{
- /* struct irda_event event; */
-
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LAP_MAGIC, return;);
- ASSERT( discovery != NULL, return;);
-
- DEBUG( 4, __FUNCTION__ "()\n");
-
- DEBUG( 4, "discovery->daddr = 0x%08x\n", discovery->daddr);
- self->daddr = discovery->daddr;
-
- /*
- * Create a new discovery log if neccessary
- */
- /* if ( self->cachelog == NULL) */
-/* self->cachelog = hashbin_new( HB_LOCAL); */
- ASSERT( self->cachelog != NULL, return;);
-
- /*
- * Insert this discovery device into the discovery_log if its
- * not there already
- */
- if ( !hashbin_find( self->cachelog, discovery->daddr, NULL))
- hashbin_insert( self->cachelog, (QUEUE *) discovery,
- discovery->daddr, NULL);
-
- irlmp_check_services( discovery);
+ irlmp_client_t *client;
+
+ DEBUG(3, __FUNCTION__ "()\n");
+
+ ASSERT(log != NULL, return;);
+
+ if (!hashbin_get_size(log))
+ return;
+
+ client = (irlmp_client_t *) hashbin_get_first(irlmp->clients);
+ while (client != NULL) {
+ /* Check if we should notify client */
+ irlmp_notify_client(client, log);
+
+ client = (irlmp_client_t *) hashbin_get_next(irlmp->clients);
+ }
}
/*
@@ -807,14 +853,13 @@ void irlmp_discovery_indication( struct lap_cb *self, DISCOVERY *discovery)
* Used by IrLAP to get the disocvery info it needs when answering
* discovery requests by other devices.
*/
-DISCOVERY *irlmp_get_discovery_response()
+discovery_t *irlmp_get_discovery_response()
{
- DEBUG( 4, "irlmp_get_discovery_response()\n");
+ DEBUG(4, __FUNCTION__ "()\n");
- ASSERT( irlmp != NULL, return NULL;);
+ ASSERT(irlmp != NULL, return NULL;);
- irlmp->discovery_rsp.hint[0] = irlmp->hint[0];
- irlmp->discovery_rsp.hint[1] = irlmp->hint[1];
+ irlmp->discovery_rsp.hints.word = irlmp->hints.word;
/*
* Set character set for device name (we use ASCII), and
@@ -823,8 +868,8 @@ DISCOVERY *irlmp_get_discovery_response()
*/
irlmp->discovery_rsp.charset = CS_ASCII;
- strncpy( irlmp->discovery_rsp.info, sysctl_devname, 31);
- irlmp->discovery_rsp.info_len = strlen( irlmp->discovery_rsp.info) + 2;
+ strncpy(irlmp->discovery_rsp.info, sysctl_devname, 31);
+ irlmp->discovery_rsp.info_len = strlen(irlmp->discovery_rsp.info) + 2;
return &irlmp->discovery_rsp;
}
@@ -835,19 +880,17 @@ DISCOVERY *irlmp_get_discovery_response()
* Send some data to peer device
*
*/
-void irlmp_data_request( struct lsap_cb *self, struct sk_buff *skb)
+void irlmp_data_request(struct lsap_cb *self, struct sk_buff *skb)
{
- DEBUG( 4, __FUNCTION__ "()\n");
-
- ASSERT( skb != NULL, return;);
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
+ ASSERT(skb != NULL, return;);
+ ASSERT(self != NULL, return;);
+ ASSERT(self->magic == LMP_LSAP_MAGIC, return;);
/* Make room for MUX header */
- ASSERT( skb_headroom( skb) >= LMP_HEADER, return;);
- skb_push( skb, LMP_HEADER);
+ ASSERT(skb_headroom( skb) >= LMP_HEADER, return;);
+ skb_push(skb, LMP_HEADER);
- irlmp_do_lsap_event( self, LM_DATA_REQUEST, skb);
+ irlmp_do_lsap_event(self, LM_DATA_REQUEST, skb);
}
/*
@@ -856,18 +899,12 @@ void irlmp_data_request( struct lsap_cb *self, struct sk_buff *skb)
* Got data from LAP layer so pass it up to upper layer
*
*/
-void irlmp_data_indication( struct lsap_cb *self, struct sk_buff *skb)
+inline void irlmp_data_indication(struct lsap_cb *self, struct sk_buff *skb)
{
- DEBUG( 4, "irlmp_data_indication()\n");
-
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
- ASSERT( skb != NULL, return;);
-
/* Hide LMP header from layer above */
- skb_pull( skb, LMP_HEADER);
+ skb_pull(skb, LMP_HEADER);
- if ( self->notify.data_indication)
+ if (self->notify.data_indication)
self->notify.data_indication(self->notify.instance, self, skb);
}
@@ -877,13 +914,11 @@ void irlmp_data_indication( struct lsap_cb *self, struct sk_buff *skb)
*
*
*/
-void irlmp_udata_request( struct lsap_cb *self, struct sk_buff *skb)
+inline void irlmp_udata_request( struct lsap_cb *self, struct sk_buff *skb)
{
DEBUG( 4, __FUNCTION__ "()\n");
ASSERT( skb != NULL, return;);
- ASSERT( self != NULL, return;);
- ASSERT( self->magic == LMP_LSAP_MAGIC, return;);
/* Make room for MUX header */
ASSERT( skb_headroom( skb) >= LMP_HEADER, return;);
@@ -909,9 +944,8 @@ void irlmp_udata_indication( struct lsap_cb *self, struct sk_buff *skb)
/* Hide LMP header from layer above */
skb_pull( skb, LMP_HEADER);
- if ( self->notify.udata_indication)
- self->notify.udata_indication( self->notify.instance, self,
- skb);
+ if (self->notify.udata_indication)
+ self->notify.udata_indication(self->notify.instance, self, skb);
}
/*
@@ -922,7 +956,7 @@ void irlmp_udata_indication( struct lsap_cb *self, struct sk_buff *skb)
*/
void irlmp_connectionless_data_request( struct sk_buff *skb)
{
- DEBUG( 0, __FUNCTION__ "(), Sorry not implemented\n");
+ DEBUG( 1, __FUNCTION__ "(), Sorry not implemented\n");
}
/*
@@ -933,12 +967,12 @@ void irlmp_connectionless_data_request( struct sk_buff *skb)
*/
void irlmp_connectionless_data_indication( struct sk_buff *skb)
{
- DEBUG( 0, __FUNCTION__ "()\n");
+ DEBUG( 1, __FUNCTION__ "()\n");
}
void irlmp_status_request(void)
{
- DEBUG( 0, "irlmp_status_request(), Not implemented\n");
+ DEBUG( 1, "irlmp_status_request(), Not implemented\n");
}
void irlmp_status_indication( LINK_STATUS link, LOCK_STATUS lock)
@@ -952,38 +986,41 @@ void irlmp_status_indication( LINK_STATUS link, LOCK_STATUS lock)
* Returns a list of all servics contained in the given hint bits. This
* funtion assumes that the hint bits have the size of two bytes only
*/
-__u8 *irlmp_hint_to_service( __u8 *hint)
+__u8 *irlmp_hint_to_service(__u8 *hint)
{
__u8 *service;
int i = 0;
- /* Allocate array to store services in */
+ /*
+ * Allocate array to store services in. 16 entries should be safe
+ * since we currently only support 2 hint bytes
+ */
service = kmalloc( 16, GFP_ATOMIC);
if ( !service) {
- DEBUG( 0, "irlmp_hint_to_service: Unable to kmalloc!\n");
+ DEBUG(1, __FUNCTION__ "(), Unable to kmalloc!\n");
return NULL;
}
if ( !hint[0]) {
- printk( "<None>\n");
+ DEBUG(1, "<None>\n");
return NULL;
}
if ( hint[0] & HINT_PNP)
- printk( "PnP Compatible ");
+ DEBUG(1, "PnP Compatible ");
if ( hint[0] & HINT_PDA)
- printk( "PDA/Palmtop ");
+ DEBUG(1, "PDA/Palmtop ");
if ( hint[0] & HINT_COMPUTER)
- printk( "Computer ");
+ DEBUG(1, "Computer ");
if ( hint[0] & HINT_PRINTER) {
- printk( "Printer\n");
+ DEBUG(1, "Printer ");
service[i++] = S_PRINTER;
}
if ( hint[0] & HINT_MODEM)
- printk( "Modem ");
+ DEBUG(1, "Modem ");
if ( hint[0] & HINT_FAX)
- printk( "Fax ");
+ DEBUG(1, "Fax ");
if ( hint[0] & HINT_LAN) {
- printk( "LAN Access\n");
+ DEBUG(1, "LAN Access ");
service[i++] = S_LAN;
}
/*
@@ -992,22 +1029,25 @@ __u8 *irlmp_hint_to_service( __u8 *hint)
* (IrLMP p. 29)
*/
if ( hint[0] & HINT_EXTENSION) {
- if ( hint[1] & HINT_TELEPHONY)
- printk( "Telephony ");
-
- if ( hint[1] & HINT_FILE_SERVER)
- printk( "File Server ");
+ if ( hint[1] & HINT_TELEPHONY) {
+ DEBUG(1, "Telephony ");
+ service[i++] = S_TELEPHONY;
+ } if ( hint[1] & HINT_FILE_SERVER)
+ DEBUG(1, "File Server ");
if ( hint[1] & HINT_COMM) {
- printk( "IrCOMM ");
+ DEBUG(1, "IrCOMM ");
service[i++] = S_COMM;
}
if ( hint[1] & HINT_OBEX) {
- printk( "IrOBEX ");
+ DEBUG(1, "IrOBEX ");
service[i++] = S_OBEX;
}
}
- printk( "\n");
+ DEBUG(1, "\n");
+
+ /* So that client can be notified about any discovery */
+ service[i++] = S_ANY;
service[i] = S_END;
@@ -1015,139 +1055,219 @@ __u8 *irlmp_hint_to_service( __u8 *hint)
}
/*
- * Function irlmp_service_to_hint (service, hint)
+ * Function irlmp_service_to_hint (service)
*
- *
+ * Converts a service type, to a hint bit
*
+ * Returns: a 16 bit hint value, with the service bit set
*/
-void irlmp_service_to_hint( int service, __u8 *hint)
+__u16 irlmp_service_to_hint(int service)
{
+ __u16_host_order hint;
+
+ hint.word = 0;
+
switch (service) {
case S_PNP:
- hint[0] |= HINT_PNP;
+ hint.byte[0] |= HINT_PNP;
break;
case S_PDA:
- hint[0] |= HINT_PDA;
+ hint.byte[0] |= HINT_PDA;
break;
case S_COMPUTER:
- hint[0] |= HINT_COMPUTER;
+ hint.byte[0] |= HINT_COMPUTER;
break;
case S_PRINTER:
- hint[0] |= HINT_PRINTER;
+ hint.byte[0] |= HINT_PRINTER;
break;
case S_MODEM:
- hint[0] |= HINT_PRINTER;
+ hint.byte[0] |= HINT_PRINTER;
break;
case S_LAN:
- hint[0] |= HINT_LAN;
+ hint.byte[0] |= HINT_LAN;
break;
case S_COMM:
- hint[0] |= HINT_EXTENSION;
- hint[1] |= HINT_COMM;
+ hint.byte[0] |= HINT_EXTENSION;
+ hint.byte[1] |= HINT_COMM;
break;
case S_OBEX:
- hint[0] |= HINT_EXTENSION;
- hint[1] |= HINT_OBEX;
+ hint.byte[0] |= HINT_EXTENSION;
+ hint.byte[1] |= HINT_OBEX;
+ break;
+ case S_ANY:
+ hint.word = 0xffff;
break;
default:
- DEBUG( 0, "irlmp_service_to_hint(), Unknown service!\n");
+ DEBUG( 1, __FUNCTION__ "(), Unknown service!\n");
break;
}
+ return hint.word;
}
/*
- * Function irlmp_register (service, type, callback)
+ * Function irlmp_register_service (service)
*
- * Register a local client or server with IrLMP
+ * Register local service with IrLMP
*
*/
-void irlmp_register_layer( int service, int type, int do_discovery,
- DISCOVERY_CALLBACK callback)
+__u32 irlmp_register_service(__u16 hints)
{
- struct irlmp_registration *entry;
+ irlmp_service_t *service;
+ __u32 handle;
- sysctl_discovery |= do_discovery;
+ DEBUG(4, __FUNCTION__ "(), hints = %04x\n", hints);
- if ( type & SERVER)
- irlmp_service_to_hint( service, irlmp->hint);
+ /* Get a unique handle for this service */
+ get_random_bytes(&handle, sizeof(handle));
+ while (hashbin_find(irlmp->services, handle, NULL) || !handle)
+ get_random_bytes(&handle, sizeof(handle));
- /* Check if this service has been registred before */
- entry = hashbin_find( irlmp->registry, service, NULL);
- if ( entry != NULL) {
- /* Update type in entry */
- entry->type |= type;
-
- /* Update callback only if client, since servers don't
- * use callbacks, and we don't want to overwrite a
- * previous registred client callback
- */
- if ( type & CLIENT)
- entry->discovery_callback = callback;
- return;
- }
+ irlmp->hints.word |= hints;
/* Make a new registration */
- entry = kmalloc( sizeof( struct irlmp_registration), GFP_ATOMIC);
- if ( !entry) {
- DEBUG( 0, "irlmp_register(), Unable to kmalloc!\n");
- return;
+ service = kmalloc(sizeof(irlmp_service_t), GFP_ATOMIC);
+ if (!service) {
+ DEBUG(1, __FUNCTION__ "(), Unable to kmalloc!\n");
+ return 0;
}
+ service->hints = hints;
+ hashbin_insert(irlmp->services, (QUEUE*) service, handle, NULL);
- entry->service = service;
- entry->type = type;
- entry->discovery_callback = callback;
-
- hashbin_insert( irlmp->registry, (QUEUE*) entry, entry->service, NULL);
+ return handle;
}
/*
- * Function irlmp_unregister (serivice)
+ * Function irlmp_unregister_service (handle)
*
- *
+ * Unregister service with IrLMP.
*
+ * Returns: 0 on success, -1 on error
*/
-void irlmp_unregister_layer( int service, int type)
+int irlmp_unregister_service(__u32 handle)
{
- struct irlmp_registration *entry;
-
- DEBUG( 4, __FUNCTION__ "()\n");
-
- entry = hashbin_find( irlmp->registry, service, NULL);
- if ( entry != NULL) {
- DEBUG( 4, "Found entry to change or remove!\n");
- /* Remove this type from the service registration */
- entry->type &= ~type;
- }
+ irlmp_service_t *service;
+
+ DEBUG(4, __FUNCTION__ "()\n");
- if ( !entry) {
- DEBUG( 0, "Unable to find entry to unregister!\n");
- return;
+ if (!handle)
+ return -1;
+
+ service = hashbin_find(irlmp->services, handle, NULL);
+ if (!service) {
+ DEBUG(1, __FUNCTION__ "(), Unknown service!\n");
+ return -1;
}
- /*
- * Remove entry if there is no more client and server support
- * left in entry
- */
- if ( !entry->type) {
- DEBUG( 4, __FUNCTION__ "(), removing entry!\n");
- entry = hashbin_remove( irlmp->registry, service, NULL);
- if ( entry != NULL)
- kfree( entry);
- }
+ service = hashbin_remove(irlmp->services, handle, NULL);
+ if (service)
+ kfree(service);
/* Remove old hint bits */
- irlmp->hint[0] = 0;
- irlmp->hint[1] = 0;
+ irlmp->hints.word = 0;
/* Refresh current hint bits */
- entry = (struct irlmp_registration *) hashbin_get_first( irlmp->registry);
- while( entry != NULL) {
- if ( entry->type & SERVER)
- irlmp_service_to_hint( entry->service,
- irlmp->hint);
- entry = (struct irlmp_registration *)
- hashbin_get_next( irlmp->registry);
+ service = (irlmp_service_t *) hashbin_get_first(irlmp->services);
+ while (service) {
+ irlmp->hints.word |= service->hints;
+
+ service = (irlmp_service_t *)hashbin_get_next(irlmp->services);
}
+ return 0;
+}
+
+/*
+ * Function irlmp_register_client (hint_mask, callback1, callback2)
+ *
+ * Register a local client with IrLMP
+ *
+ * Returns: handle > 0 on success, 0 on error
+ */
+__u32 irlmp_register_client(__u16 hint_mask, DISCOVERY_CALLBACK1 callback1,
+ DISCOVERY_CALLBACK2 callback2)
+{
+ irlmp_client_t *client;
+ __u32 handle;
+
+ /* Get a unique handle for this client */
+ get_random_bytes(&handle, sizeof(handle));
+ while (hashbin_find(irlmp->clients, handle, NULL) || !handle)
+ get_random_bytes(&handle, sizeof(handle));
+
+ /* Make a new registration */
+ client = kmalloc(sizeof(irlmp_client_t), GFP_ATOMIC);
+ if (!client) {
+ DEBUG( 1, __FUNCTION__ "(), Unable to kmalloc!\n");
+
+ return 0;
+ }
+
+ /* Register the details */
+ client->hint_mask = hint_mask;
+ client->callback1 = callback1;
+ client->callback2 = callback2;
+
+ hashbin_insert(irlmp->clients, (QUEUE *) client, handle, NULL);
+
+ return handle;
+}
+
+/*
+ * Function irlmp_update_client (handle, hint_mask, callback1, callback2)
+ *
+ * Updates specified client (handle) with possibly new hint_mask and
+ * callback
+ *
+ * Returns: 0 on success, -1 on error
+ */
+int irlmp_update_client(__u32 handle, __u16 hint_mask,
+ DISCOVERY_CALLBACK1 callback1,
+ DISCOVERY_CALLBACK2 callback2)
+{
+ irlmp_client_t *client;
+
+ if (!handle)
+ return -1;
+
+ client = hashbin_find(irlmp->clients, handle, NULL);
+ if (!client) {
+ DEBUG(1, __FUNCTION__ "(), Unknown client!\n");
+ return -1;
+ }
+
+ client->hint_mask = hint_mask;
+ client->callback1 = callback1;
+ client->callback2 = callback2;
+
+ return 0;
+}
+
+/*
+ * Function irlmp_unregister_client (handle)
+ *
+ * Returns: 0 on success, -1 on error
+ *
+ */
+int irlmp_unregister_client(__u32 handle)
+{
+ struct irlmp_client *client;
+
+ DEBUG(4, __FUNCTION__ "()\n");
+
+ if (!handle)
+ return -1;
+
+ client = hashbin_find(irlmp->clients, handle, NULL);
+ if (!client) {
+ DEBUG(1, __FUNCTION__ "(), Unknown client!\n");
+ return -1;
+ }
+
+ DEBUG( 4, __FUNCTION__ "(), removing client!\n");
+ client = hashbin_remove( irlmp->clients, handle, NULL);
+ if (client)
+ kfree(client);
+
+ return 0;
}
/*
@@ -1164,22 +1284,26 @@ int irlmp_slsap_inuse( __u8 slsap_sel)
ASSERT( irlmp->magic == LMP_MAGIC, return TRUE;);
ASSERT( slsap_sel != LSAP_ANY, return TRUE;);
- DEBUG( 4, "irlmp_slsap_inuse()\n");
+ DEBUG( 4, __FUNCTION__ "()\n");
+
+ /* Valid values are between 0 and 127 */
+ if (slsap_sel > 127)
+ return TRUE;
/*
* Check if slsap is already in use. To do this we have to loop over
* every IrLAP connection and check every LSAP assosiated with each
* the connection.
*/
- lap = ( struct lap_cb *) hashbin_get_first( irlmp->links);
- while ( lap != NULL) {
- ASSERT( lap->magic == LMP_LAP_MAGIC, return TRUE;);
+ lap = (struct lap_cb *) hashbin_get_first(irlmp->links);
+ while (lap != NULL) {
+ ASSERT(lap->magic == LMP_LAP_MAGIC, return TRUE;);
- self = (struct lsap_cb *) hashbin_get_first( lap->lsaps);
- while ( self != NULL) {
- ASSERT( self->magic == LMP_LSAP_MAGIC, return TRUE;);
+ self = (struct lsap_cb *) hashbin_get_first(lap->lsaps);
+ while (self != NULL) {
+ ASSERT(self->magic == LMP_LSAP_MAGIC, return TRUE;);
- if (( self->slsap_sel == slsap_sel))/* && */
+ if ((self->slsap_sel == slsap_sel))/* && */
/* ( self->dlsap_sel == LSAP_ANY)) */
{
DEBUG( 4, "Source LSAP selector=%02x in use\n",
@@ -1202,15 +1326,28 @@ int irlmp_slsap_inuse( __u8 slsap_sel)
__u8 irlmp_find_free_slsap(void)
{
__u8 lsap_sel;
+ int wrapped = 0;
- ASSERT( irlmp != NULL, return -1;);
- ASSERT( irlmp->magic == LMP_MAGIC, return -1;);
+ ASSERT(irlmp != NULL, return -1;);
+ ASSERT(irlmp->magic == LMP_MAGIC, return -1;);
lsap_sel = irlmp->free_lsap_sel++;
+
+ /* Check if the new free lsap is really free */
+ while (irlmp_slsap_inuse(irlmp->free_lsap_sel)) {
+ irlmp->free_lsap_sel++;
- DEBUG( 4, "irlmp_find_free_slsap(), picked next free lsap_sel=%02x\n",
- lsap_sel);
+ /* Check if we need to wraparound */
+ if (irlmp->free_lsap_sel > 127) {
+ irlmp->free_lsap_sel = 10;
+ /* Make sure we terminate the loop */
+ if (wrapped++)
+ return 0;
+ }
+ }
+ DEBUG(4, __FUNCTION__ "(), next free lsap_sel=%02x\n", lsap_sel);
+
return lsap_sel;
}
@@ -1227,21 +1364,25 @@ LM_REASON irlmp_convert_lap_reason( LAP_REASON lap_reason)
switch (lap_reason) {
case LAP_DISC_INDICATION: /* Received a disconnect request from peer */
+ DEBUG( 1, __FUNCTION__ "(), LAP_DISC_INDICATION\n");
reason = LM_USER_REQUEST;
break;
case LAP_NO_RESPONSE: /* To many retransmits without response */
+ DEBUG( 1, __FUNCTION__ "(), LAP_NO_RESPONSE\n");
reason = LM_LAP_DISCONNECT;
break;
case LAP_RESET_INDICATION:
+ DEBUG( 1, __FUNCTION__ "(), LAP_RESET_INDICATION\n");
reason = LM_LAP_RESET;
break;
case LAP_FOUND_NONE:
case LAP_MEDIA_BUSY:
case LAP_PRIMARY_CONFLICT:
+ DEBUG( 1, __FUNCTION__ "(), LAP_FOUND_NONE, LAP_MEDIA_BUSY or LAP_PRIMARY_CONFLICT\n");
reason = LM_CONNECT_FAILURE;
break;
default:
- DEBUG( 0, __FUNCTION__
+ DEBUG( 1, __FUNCTION__
"(), Unknow IrLAP disconnect reason %d!\n", lap_reason);
reason = LM_LAP_DISCONNECT;
break;
@@ -1250,6 +1391,26 @@ LM_REASON irlmp_convert_lap_reason( LAP_REASON lap_reason)
return reason;
}
+__u32 irlmp_get_saddr(struct lsap_cb *self)
+{
+ DEBUG(3, __FUNCTION__ "()\n");
+
+ ASSERT(self != NULL, return 0;);
+ ASSERT(self->lap != NULL, return 0;);
+
+ return self->lap->saddr;
+}
+
+__u32 irlmp_get_daddr(struct lsap_cb *self)
+{
+ DEBUG(3, __FUNCTION__ "()\n");
+
+ ASSERT(self != NULL, return 0;);
+ ASSERT(self->lap != NULL, return 0;);
+
+ return self->lap->daddr;
+}
+
#ifdef CONFIG_PROC_FS
/*
* Function irlmp_proc_read (buf, start, offset, len, unused)
@@ -1295,8 +1456,7 @@ int irlmp_proc_read( char *buf, char **start, off_t offset, int len,
len += sprintf( buf+len, "lap state: %s, ",
irlmp_state[ lap->lap_state]);
- len += sprintf( buf+len,
- "saddr: %#08x, daddr: %#08x, ",
+ len += sprintf( buf+len, "saddr: %#08x, daddr: %#08x, ",
lap->saddr, lap->daddr);
len += sprintf( buf+len, "\n");