Automatic merge of rsync://rsync.kernel.org/pub/scm/linux/kernel/git/gregkh/driver...
[linux-2.6/verdex.git] / arch / ppc64 / kernel / HvLpEvent.c
blobf8f19637f73f1fa9e0ecc514989c4e005c2c9b77
1 /*
2 * Copyright 2001 Mike Corrigan IBM Corp
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; either version
7 * 2 of the License, or (at your option) any later version.
8 */
9 #include <linux/stddef.h>
10 #include <linux/kernel.h>
11 #include <linux/module.h>
12 #include <asm/system.h>
13 #include <asm/iSeries/HvLpEvent.h>
14 #include <asm/iSeries/HvCallEvent.h>
15 #include <asm/iSeries/LparData.h>
17 /* Array of LpEvent handler functions */
18 LpEventHandler lpEventHandler[HvLpEvent_Type_NumTypes];
19 unsigned lpEventHandlerPaths[HvLpEvent_Type_NumTypes];
21 /* Register a handler for an LpEvent type */
23 int HvLpEvent_registerHandler( HvLpEvent_Type eventType, LpEventHandler handler )
25 int rc = 1;
26 if ( eventType < HvLpEvent_Type_NumTypes ) {
27 lpEventHandler[eventType] = handler;
28 rc = 0;
30 return rc;
34 int HvLpEvent_unregisterHandler( HvLpEvent_Type eventType )
36 int rc = 1;
38 might_sleep();
40 if ( eventType < HvLpEvent_Type_NumTypes ) {
41 if ( !lpEventHandlerPaths[eventType] ) {
42 lpEventHandler[eventType] = NULL;
43 rc = 0;
45 /* We now sleep until all other CPUs have scheduled. This ensures that
46 * the deletion is seen by all other CPUs, and that the deleted handler
47 * isn't still running on another CPU when we return. */
48 synchronize_rcu();
51 return rc;
53 EXPORT_SYMBOL(HvLpEvent_registerHandler);
54 EXPORT_SYMBOL(HvLpEvent_unregisterHandler);
56 /* (lpIndex is the partition index of the target partition.
57 * needed only for VirtualIo, VirtualLan and SessionMgr. Zero
58 * indicates to use our partition index - for the other types)
60 int HvLpEvent_openPath( HvLpEvent_Type eventType, HvLpIndex lpIndex )
62 int rc = 1;
63 if ( eventType < HvLpEvent_Type_NumTypes &&
64 lpEventHandler[eventType] ) {
65 if ( lpIndex == 0 )
66 lpIndex = itLpNaca.xLpIndex;
67 HvCallEvent_openLpEventPath( lpIndex, eventType );
68 ++lpEventHandlerPaths[eventType];
69 rc = 0;
71 return rc;
74 int HvLpEvent_closePath( HvLpEvent_Type eventType, HvLpIndex lpIndex )
76 int rc = 1;
77 if ( eventType < HvLpEvent_Type_NumTypes &&
78 lpEventHandler[eventType] &&
79 lpEventHandlerPaths[eventType] ) {
80 if ( lpIndex == 0 )
81 lpIndex = itLpNaca.xLpIndex;
82 HvCallEvent_closeLpEventPath( lpIndex, eventType );
83 --lpEventHandlerPaths[eventType];
84 rc = 0;
86 return rc;