patch-2.2.16 linux/arch/s390/kernel/s390dyn.c
Next file: linux/arch/s390/kernel/s390io.c
Previous file: linux/arch/s390/kernel/s390_ksyms.c
Back to the patch index
Back to the overall index
- Lines: 207
- Date:
Wed Jun 7 14:26:42 2000
- Orig file:
v2.2.15/linux/arch/s390/kernel/s390dyn.c
- Orig date:
Wed Dec 31 16:00:00 1969
diff -urN v2.2.15/linux/arch/s390/kernel/s390dyn.c linux/arch/s390/kernel/s390dyn.c
@@ -0,0 +1,206 @@
+/*
+ * arch/s390/kernel/s390dyn.c
+ * S/390 dynamic device attachment
+ *
+ * S390 version
+ * Copyright (C) 1999,2000 IBM Deutschland Entwicklung GmbH, IBM Corporation
+ * Author(s): Ingo Adlung (adlung@de.ibm.com)
+ */
+
+#include <linux/init.h>
+#include <linux/smp_lock.h>
+
+#include <asm/irq.h>
+#include <asm/s390io.h>
+#include <asm/s390dyn.h>
+
+static devreg_t *devreg_anchor = NULL;
+static spinlock_t dyn_lock = SPIN_LOCK_UNLOCKED;
+
+int s390_device_register( devreg_t *drinfo )
+{
+ unsigned long flags;
+
+ int ret = 0;
+ devreg_t *pdevreg = devreg_anchor;
+
+ if ( drinfo == NULL )
+ return( -EINVAL );
+
+ spin_lock_irqsave( &dyn_lock, flags );
+
+ while ( (pdevreg != NULL) && (ret ==0) )
+ {
+ if ( pdevreg == drinfo )
+ {
+ ret = -EINVAL;
+ }
+ else
+ {
+ if ( ( (pdevreg->flag & DEVREG_TYPE_DEVNO)
+ && (pdevreg->ci.devno ) )
+ && ( (drinfo->flag & DEVREG_TYPE_DEVNO )
+ && (drinfo->ci.devno ) ) )
+ {
+ ret = -EBUSY;
+ }
+ else if ( (pdevreg->flag & DEVREG_EXACT_MATCH)
+ && (drinfo->flag & DEVREG_EXACT_MATCH ) )
+ {
+ if ( memcmp( &pdevreg->ci.hc, &drinfo->ci.hc, 6) )
+ ret = -EBUSY;
+ }
+ else if ( (pdevreg->flag & DEVREG_MATCH_DEV_TYPE)
+ && (drinfo->flag & DEVREG_MATCH_DEV_TYPE ) )
+ {
+ if ( (pdevreg->ci.hc.dtype == drinfo->ci.hc.dtype)
+ && (pdevreg->ci.hc.dmode == drinfo->ci.hc.dmode) )
+ ret = -EBUSY;
+ }
+ else if ( (pdevreg->flag & DEVREG_MATCH_CU_TYPE)
+ && (drinfo->flag & DEVREG_MATCH_CU_TYPE ) )
+ {
+ if ( (pdevreg->ci.hc.ctype == drinfo->ci.hc.ctype)
+ && (pdevreg->ci.hc.cmode == drinfo->ci.hc.cmode) )
+ ret = -EBUSY;
+ }
+ else if ( (pdevreg->flag & DEVREG_NO_CU_INFO)
+ && (drinfo->flag & DEVREG_NO_CU_INFO ) )
+ {
+ if ( (pdevreg->ci.hnc.dtype == drinfo->ci.hnc.dtype)
+ && (pdevreg->ci.hnc.dmode == drinfo->ci.hnc.dmode) )
+ ret = -EBUSY;
+ }
+
+ pdevreg = pdevreg->next;
+
+ } /* endif */
+
+ } /* endwhile */
+
+ /*
+ * only enqueue if no collision was found ...
+ */
+ if ( ret == 0 )
+ {
+ drinfo->next = devreg_anchor;
+ drinfo->prev = NULL;
+
+ if ( devreg_anchor != NULL )
+ {
+ devreg_anchor->prev = drinfo;
+
+ } /* endif */
+
+ } /* endif */
+
+ spin_unlock_irqrestore( &dyn_lock, flags );
+
+ return( ret);
+}
+
+
+int s390_device_unregister( devreg_t *dreg )
+{
+ unsigned long flags;
+
+ int ret = -EINVAL;
+ devreg_t *pdevreg = devreg_anchor;
+
+ if ( dreg == NULL )
+ return( -EINVAL );
+
+ spin_lock_irqsave( &dyn_lock, flags );
+
+ while ( (pdevreg != NULL )
+ && ( ret != 0 ) )
+ {
+ if ( pdevreg == dreg )
+ {
+ devreg_t *dprev = pdevreg->prev;
+ devreg_t *dnext = pdevreg->next;
+
+ if ( (dprev != NULL) && (dnext != NULL) )
+ {
+ dnext->prev = dprev;
+ dprev->next = dnext;
+ }
+ if ( (dprev != NULL) && (dnext == NULL) )
+ {
+ dprev->next = NULL;
+ }
+ if ( (dprev == NULL) && (dnext != NULL) )
+ {
+ dnext->prev = NULL;
+
+ } /* else */
+
+ ret = 0;
+ }
+ else
+ {
+ pdevreg = pdevreg->next;
+
+ } /* endif */
+
+ } /* endwhile */
+
+ spin_unlock_irqrestore( &dyn_lock, flags );
+
+ return( ret);
+}
+
+
+devreg_t * s390_search_devreg( ioinfo_t *ioinfo )
+{
+ unsigned long flags;
+
+ devreg_t *pdevreg = devreg_anchor;
+
+ if ( ioinfo == NULL )
+ return( NULL );
+
+ spin_lock_irqsave( &dyn_lock, flags );
+
+ while ( pdevreg != NULL )
+ {
+ if ( (pdevreg->flag & DEVREG_TYPE_DEVNO )
+ && (ioinfo->ui.flags.dval == 1 )
+ && (ioinfo->devno == pdevreg->ci.devno) )
+ {
+ break;
+ }
+ else if ( pdevreg->flag & DEVREG_EXACT_MATCH )
+ {
+ if ( memcmp( &pdevreg->ci.hc,
+ &ioinfo->senseid.cu_type, 6 ) )
+ break;
+ }
+ else if ( pdevreg->flag & DEVREG_MATCH_DEV_TYPE )
+ {
+ if ( (pdevreg->ci.hc.dtype == ioinfo->senseid.dev_type )
+ && (pdevreg->ci.hc.dmode == ioinfo->senseid.dev_model) )
+ break;
+ }
+ else if ( pdevreg->flag & DEVREG_MATCH_CU_TYPE )
+ {
+ if ( (pdevreg->ci.hc.ctype == ioinfo->senseid.cu_type )
+ && (pdevreg->ci.hc.cmode == ioinfo->senseid.cu_model) )
+ break;
+ }
+ else if ( pdevreg->flag & DEVREG_NO_CU_INFO )
+ {
+ if ( (pdevreg->ci.hnc.dtype == ioinfo->senseid.dev_type )
+ && (pdevreg->ci.hnc.dmode == ioinfo->senseid.dev_model) )
+ break;
+ }
+
+ pdevreg = pdevreg->next;
+
+ } /* endwhile */
+
+ spin_unlock_irqrestore( &dyn_lock, flags );
+
+ return( pdevreg);
+}
+
FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen (who was at: slshen@lbl.gov)