patch-2.1.132 linux/net/irda/irmod.c
Next file: linux/net/irda/irobex/Config.in
Previous file: linux/net/irda/irlmp_frame.c
Back to the patch index
Back to the overall index
- Lines: 398
- Date:
Thu Dec 17 09:01:03 1998
- Orig file:
v2.1.131/linux/net/irda/irmod.c
- Orig date:
Wed Dec 31 16:00:00 1969
diff -u --recursive --new-file v2.1.131/linux/net/irda/irmod.c linux/net/irda/irmod.c
@@ -0,0 +1,397 @@
+/*********************************************************************
+ *
+ * Filename: irmod.c
+ * Version: 0.8
+ * Description: IrDA module code and some other stuff
+ * Status: Experimental.
+ * Author: Dag Brattli <dagb@cs.uit.no>
+ * Created at: Mon Dec 15 13:55:39 1997
+ * Modified at: Mon Dec 14 20:10:28 1998
+ * Modified by: Dag Brattli <dagb@cs.uit.no>
+ *
+ * Copyright (c) 1997 Dag Brattli, All Rights Reserved.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of
+ * the License, or (at your option) any later version.
+ *
+ * Neither Dag Brattli nor University of Tromsų admit liability nor
+ * provide warranty for any of this software. This material is
+ * provided "AS-IS" and at no charge.
+ *
+ ********************************************************************/
+
+#include <linux/config.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <asm/segment.h>
+#include <linux/poll.h>
+
+#include <net/irda/irda.h>
+#include <net/irda/irlmp.h>
+#include <net/irda/iriap.h>
+#include <net/irda/irttp.h>
+
+struct irda irda; /* One global instance */
+
+extern void irda_proc_register(void);
+extern void irda_proc_unregister(void);
+extern int irda_sysctl_register(void);
+extern void irda_sysctl_unregister(void);
+
+extern void irda_proto_init(struct net_proto *pro);
+extern void irda_proto_cleanup(void);
+
+extern int irda_device_init(void);
+extern int irobex_init(void);
+extern int irlan_init(void);
+extern int irlan_client_init(void);
+extern int irlan_server_init(void);
+extern int ircomm_init(void);
+extern int irvtd_init(void);
+
+static int irda_open( struct inode * inode, struct file *file);
+static int irda_ioctl( struct inode *inode, struct file *filp,
+ unsigned int cmd, unsigned long arg);
+static int irda_close( struct inode *inode, struct file *file);
+static ssize_t irda_read( struct file *file, char *buffer, size_t count,
+ loff_t *noidea);
+static ssize_t irda_write( struct file *file, const char *buffer,
+ size_t count, loff_t *noidea);
+static u_int irda_poll( struct file *file, poll_table *wait);
+
+static struct file_operations irda_fops = {
+ NULL, /* seek */
+ irda_read, /* read */
+ irda_write, /* write */
+ NULL, /* readdir */
+ irda_poll, /* poll */
+ irda_ioctl, /* ioctl */
+ NULL, /* mmap */
+ irda_open,
+ NULL,
+ irda_close,
+ NULL,
+ NULL, /* fasync */
+};
+
+__initfunc(int irda_init(void))
+{
+ printk( KERN_INFO "Linux Support for the IrDA (tm) protocols (Dag Brattli)\n");
+
+ irda_device_init();
+ irlap_init();
+ irlmp_init();
+ iriap_init();
+ irttp_init();
+
+#ifdef CONFIG_PROC_FS
+ irda_proc_register();
+#endif
+#ifdef CONFIG_SYSCTL
+ irda_sysctl_register();
+#endif
+
+ irda.dev.minor = MISC_DYNAMIC_MINOR;
+ irda.dev.name = "irda";
+ irda.dev.fops = &irda_fops;
+
+ misc_register( &irda.dev);
+
+ /*
+ * Initialize modules that got compiled into the kernel
+ */
+#ifdef CONFIG_IRLAN
+ irlan_init();
+#endif
+#ifdef CONFIG_IRLAN_CLIENT
+ irlan_client_init();
+#endif
+#ifdef CONFIG_IRLAN_SERVER
+ irlan_server_init();
+#endif
+#ifdef CONFIG_IROBEX
+ irobex_init();
+#endif
+#ifdef CONFIG_IRCOMM
+ ircomm_init();
+ irvtd_init();
+#endif
+
+ return 0;
+}
+
+void irda_cleanup(void)
+{
+ misc_deregister( &irda.dev);
+
+#ifdef CONFIG_SYSCTL
+ irda_sysctl_unregister();
+#endif
+
+#ifdef CONFIG_PROC_FS
+ irda_proc_unregister();
+#endif
+
+ /* Remove higher layers */
+ irttp_cleanup();
+ iriap_cleanup();
+
+ /* Remove lower layers */
+ irda_device_cleanup();
+ irlap_cleanup(); /* Must be done before irlmp_cleanup()! DB */
+
+ /* Remove middle layer */
+ irlmp_cleanup();
+}
+
+/*
+ * Function irda_lock (lock)
+ *
+ * Lock variable. Returns false if the lock is already set.
+ *
+ */
+inline int irda_lock( int *lock) {
+ if ( test_and_set_bit( 0, (void *) lock)) {
+ printk("Trying to lock, already locked variable!\n");
+ return FALSE;
+ }
+ return TRUE;
+}
+
+/*
+ * Function irda_unlock (lock)
+ *
+ * Unlock variable. Returns false if lock is already unlocked
+ *
+ */
+inline int irda_unlock( int *lock) {
+ if ( !test_and_clear_bit( 0, (void *) lock)) {
+ printk("Trying to unlock already unlocked variable!\n");
+ return FALSE;
+ }
+ return TRUE;
+}
+
+/*
+ * Function irda_notify_init (notify)
+ *
+ * Used for initializing the notify structure
+ *
+ */
+void irda_notify_init( struct notify_t *notify)
+{
+
+ notify->data_indication = NULL;
+ notify->udata_indication = NULL;
+ notify->connect_confirm = NULL;
+ notify->connect_indication = NULL;
+ notify->disconnect_indication = NULL;
+ notify->flow_indication = NULL;
+ notify->instance = NULL;
+ strncpy( notify->name, "Unknown", NOTIFY_MAX_NAME);
+}
+
+/*
+ * Function irda_execute_as_process (self, callback, param)
+ *
+ * If a layer needs to have a function executed with a process context,
+ * then it can register the function here, and the function will then
+ * be executed as fast as possible.
+ *
+ */
+void irda_execute_as_process( void *self, TODO_CALLBACK callback, __u32 param)
+{
+ struct irda_todo *new;
+ struct irmanager_event event;
+
+ /* Make new todo event */
+ new = (struct irda_todo *) kmalloc( sizeof(struct irda_todo),
+ GFP_ATOMIC);
+ if ( new == NULL) {
+ return;
+ }
+ memset( new, 0, sizeof( struct irda_todo));
+
+ new->self = self;
+ new->callback = callback;
+ new->param = param;
+
+ /* Queue todo */
+ enqueue_last( &irda.todo_queue, (QUEUE *) new);
+
+ event.event = EVENT_NEED_PROCESS_CONTEXT;
+
+ /* Notify the user space manager */
+ irmanager_notify( &event);
+}
+
+/*
+ * Function irmanger_notify (event)
+ *
+ * Send an event to the user space manager
+ *
+ */
+void irmanager_notify( struct irmanager_event *event)
+{
+ struct irda_event *new;
+
+ DEBUG( 4, __FUNCTION__ "()\n");
+
+ /* Make new IrDA Event */
+ new = (struct irda_event *) kmalloc( sizeof(struct irda_event),
+ GFP_ATOMIC);
+ if ( new == NULL) {
+ return;
+ }
+ memset( new, 0, sizeof( struct irda_event));
+ new->event = *event;
+
+ /* Queue event */
+ enqueue_last( &irda.event_queue, (QUEUE *) new);
+
+ /* Wake up irmanager sleeping on read */
+ wake_up_interruptible( &irda.wait_queue);
+}
+
+static int irda_open( struct inode * inode, struct file *file)
+{
+ DEBUG( 4, __FUNCTION__ "()\n");
+
+ MOD_INC_USE_COUNT;
+
+ return 0;
+}
+
+/*
+ * Function irda_ioctl (inode, filp, cmd, arg)
+ *
+ * Ioctl, used by irmanager to ...
+ *
+ */
+static int irda_ioctl( struct inode *inode, struct file *filp,
+ unsigned int cmd, unsigned long arg)
+{
+ struct irda_todo *todo;
+ int err = 0;
+ int size = _IOC_SIZE(cmd);
+
+ DEBUG( 4, __FUNCTION__ "()\n");
+
+ if ( _IOC_DIR(cmd) & _IOC_READ)
+ err = verify_area( VERIFY_WRITE, (void *) arg, size);
+ else if ( _IOC_DIR(cmd) & _IOC_WRITE)
+ err = verify_area( VERIFY_READ, (void *) arg, size);
+ if ( err)
+ return err;
+
+ switch( cmd) {
+ case IRMGR_IOCTNPC:
+ /* Got process context! */
+ DEBUG( 4, __FUNCTION__ "(), got process context!\n");
+
+ while (( todo = (struct irda_todo *) dequeue_first(
+ &irda.todo_queue)) != NULL)
+ {
+ todo->callback( todo->self, todo->param);
+
+ kfree( todo);
+ }
+ break;
+
+ default:
+ return -ENOIOCTLCMD;
+ }
+
+ return 0;
+}
+
+static int irda_close( struct inode *inode, struct file *file)
+{
+ DEBUG( 4, __FUNCTION__ "()\n");
+
+ MOD_DEC_USE_COUNT;
+
+ return 0;
+}
+
+static ssize_t irda_read( struct file *file, char *buffer, size_t count,
+ loff_t *noidea)
+{
+ struct irda_event *event;
+ unsigned long flags;
+ int len;
+
+ DEBUG( 4, __FUNCTION__ "()\n");
+
+ /* * Go to sleep and wait for event if there is no event to be read! */
+ save_flags( flags);
+ cli();
+ if ( !irda.event_queue)
+ interruptible_sleep_on( &irda.wait_queue);
+ restore_flags(flags);
+
+ /*
+ * Ensure proper reaction to signals, and screen out
+ * blocked signals (page 112. linux device drivers)
+ */
+ if ( signal_pending( current))
+ return -ERESTARTSYS;
+
+ event = (struct irda_event *) dequeue_first( &irda.event_queue);
+
+ len = sizeof(struct irmanager_event);
+ copy_to_user( buffer, &event->event, len);
+
+ /* Finished with event */
+ kfree( event);
+
+ return len;
+}
+
+static ssize_t irda_write( struct file *file, const char *buffer,
+ size_t count, loff_t *noidea)
+{
+ DEBUG( 0, __FUNCTION__ "()\n");
+
+ return 0;
+}
+
+static u_int irda_poll( struct file *file, poll_table *wait)
+{
+ DEBUG( 0, __FUNCTION__ "(), Sorry not implemented yet!\n");
+
+ return 0;
+}
+
+#ifdef MODULE
+
+MODULE_AUTHOR("Dag Brattli <dagb@cs.uit.no>");
+MODULE_DESCRIPTION("The Linux IrDA protocol subsystem");
+
+/*
+ * Function init_module (void)
+ *
+ * Initialize the irda module
+ *
+ */
+int init_module(void)
+{
+ irda_proto_init(NULL);
+
+ return 0;
+}
+
+/*
+ * Function cleanup_module (void)
+ *
+ * Cleanup the irda module
+ *
+ */
+void cleanup_module(void)
+{
+ irda_proto_cleanup();
+}
+
+#endif
FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen, slshen@lbl.gov