Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
004ffedc
Commit
004ffedc
authored
Sep 29, 2002
by
Arnaldo Carvalho de Melo
Committed by
David S. Miller
Sep 29, 2002
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
o lapbether: get rid of cli/sti, use refcnts for devs, etc
parent
a134be24
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
168 additions
and
137 deletions
+168
-137
drivers/net/wan/lapbether.c
drivers/net/wan/lapbether.c
+168
-137
No files found.
drivers/net/wan/lapbether.c
View file @
004ffedc
...
@@ -44,42 +44,63 @@
...
@@ -44,42 +44,63 @@
#include <linux/lapb.h>
#include <linux/lapb.h>
#include <linux/init.h>
#include <linux/init.h>
static
char
bcast_addr
[
6
]
=
{
0xFF
,
0xFF
,
0xFF
,
0xFF
,
0xFF
,
0xFF
};
static
char
bcast_addr
[
6
]
=
{
0xFF
,
0xFF
,
0xFF
,
0xFF
,
0xFF
,
0xFF
};
/* If this number is made larger, check that the temporary string buffer
/* If this number is made larger, check that the temporary string buffer
* in lapbeth_new_device is large enough to store the probe device name.*/
* in lapbeth_new_device is large enough to store the probe device name.*/
#define MAXLAPBDEV 100
#define MAXLAPBDEV 100
static
struct
lapbethdev
{
struct
lapbethdev
{
struct
lapbethdev
*
next
;
struct
list_head
node
;
char
ethname
[
14
];
/* ether device name */
char
ethname
[
14
];
/* ether device name */
struct
net_device
*
ethdev
;
/* link to ethernet device */
struct
net_device
*
ethdev
;
/* link to ethernet device */
struct
net_device
axdev
;
/* lapbeth device (lapb#) */
struct
net_device
axdev
;
/* lapbeth device (lapb#) */
struct
net_device_stats
stats
;
/* some statistics */
struct
net_device_stats
stats
;
/* some statistics */
}
*
lapbeth_devices
/* = NULL initially */
;
atomic_t
refcnt
;
};
static
struct
list_head
lapbeth_devices
=
LIST_HEAD_INIT
(
lapbeth_devices
);
static
rwlock_t
lapbeth_devices_lock
=
RW_LOCK_UNLOCKED
;
static
__inline__
void
lapbeth_hold
(
struct
lapbethdev
*
lapbeth
)
{
atomic_inc
(
&
lapbeth
->
refcnt
);
}
static
__inline__
void
lapbeth_put
(
struct
lapbethdev
*
lapbeth
)
{
if
(
atomic_dec_and_test
(
&
lapbeth
->
refcnt
))
kfree
(
lapbeth
);
}
/* ------------------------------------------------------------------------ */
/* ------------------------------------------------------------------------ */
/*
/*
* Get the LAPB device for the ethernet device
* Get the LAPB device for the ethernet device
*/
*/
static
inline
struct
net_device
*
lapbeth_get_x25_dev
(
struct
net_device
*
dev
)
static
__inline__
struct
lapbethdev
*
lapbeth_get_x25_dev
(
struct
net_device
*
dev
)
{
{
struct
lapbethdev
*
lapbeth
;
struct
list_head
*
entry
;
struct
lapbethdev
*
lapbeth
,
*
use
=
NULL
;
read_lock
(
&
lapbeth_devices_lock
);
for
(
lapbeth
=
lapbeth_devices
;
lapbeth
!=
NULL
;
lapbeth
=
lapbeth
->
next
)
list_for_each
(
entry
,
&
lapbeth_devices
)
{
if
(
lapbeth
->
ethdev
==
dev
)
lapbeth
=
list_entry
(
entry
,
struct
lapbethdev
,
node
);
return
&
lapbeth
->
axdev
;
if
(
lapbeth
->
ethdev
==
dev
)
{
use
=
lapbeth
;
break
;
}
}
if
(
use
)
lapbeth_hold
(
use
);
return
NULL
;
read_unlock
(
&
lapbeth_devices_lock
);
return
use
;
}
}
static
inline
int
dev_is_ethdev
(
struct
net_device
*
dev
)
static
__inline__
int
dev_is_ethdev
(
struct
net_device
*
dev
)
{
{
return
(
return
dev
->
type
==
ARPHRD_ETHER
&&
strncmp
(
dev
->
name
,
"dummy"
,
5
);
dev
->
type
==
ARPHRD_ETHER
&&
strncmp
(
dev
->
name
,
"dummy"
,
5
)
);
}
}
/*
/*
...
@@ -88,35 +109,26 @@ static inline int dev_is_ethdev(struct net_device *dev)
...
@@ -88,35 +109,26 @@ static inline int dev_is_ethdev(struct net_device *dev)
*/
*/
static
int
lapbeth_check_devices
(
struct
net_device
*
dev
)
static
int
lapbeth_check_devices
(
struct
net_device
*
dev
)
{
{
struct
lapbethdev
*
lapbeth
,
*
lapbeth_prev
,
*
lapbeth_next
;
struct
lapbethdev
*
lapbeth
;
struct
list_head
*
entry
,
*
tmp
;
int
result
=
0
;
int
result
=
0
;
unsigned
long
flags
;
save_flags
(
flags
);
write_lock
(
&
lapbeth_devices_lock
);
cli
();
lapbeth_prev
=
NULL
;
list_for_each_safe
(
entry
,
tmp
,
&
lapbeth_devices
)
{
lapbeth
=
list_entry
(
entry
,
struct
lapbethdev
,
node
);
for
(
lapbeth
=
lapbeth_devices
;
lapbeth
!=
NULL
;
lapbeth
=
lapbeth_next
)
{
lapbeth_next
=
lapbeth
->
next
;
if
(
!
dev_get
(
lapbeth
->
ethname
))
{
if
(
!
dev_get
(
lapbeth
->
ethname
))
{
if
(
lapbeth_prev
)
lapbeth_prev
->
next
=
lapbeth
->
next
;
else
lapbeth_devices
=
lapbeth
->
next
;
if
(
&
lapbeth
->
axdev
==
dev
)
if
(
&
lapbeth
->
axdev
==
dev
)
result
=
1
;
result
=
1
;
unregister_netdev
(
&
lapbeth
->
axdev
);
unregister_netdev
(
&
lapbeth
->
axdev
);
dev_put
(
lapbeth
->
ethdev
);
dev_put
(
lapbeth
->
ethdev
);
kfree
(
lapbeth
);
list_del
(
&
lapbeth
->
node
);
lapbeth_put
(
lapbeth
);
}
}
else
lapbeth_prev
=
lapbeth
;
}
}
write_unlock
(
&
lapbeth_devices_lock
);
restore_flags
(
flags
);
return
result
;
return
result
;
}
}
...
@@ -133,14 +145,12 @@ static int lapbeth_rcv(struct sk_buff *skb, struct net_device *dev, struct packe
...
@@ -133,14 +145,12 @@ static int lapbeth_rcv(struct sk_buff *skb, struct net_device *dev, struct packe
skb
->
sk
=
NULL
;
/* Initially we don't know who it's for */
skb
->
sk
=
NULL
;
/* Initially we don't know who it's for */
dev
=
lapbeth_get_x25_dev
(
dev
);
lapbeth
=
lapbeth_get_x25_dev
(
dev
);
if
(
dev
==
NULL
||
!
netif_running
(
dev
))
{
if
(
!
lapbeth
)
kfree_skb
(
skb
);
goto
drop
;
return
0
;
if
(
!
netif_running
(
&
lapbeth
->
axdev
))
}
goto
put_drop
;
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
lapbeth
->
stats
.
rx_packets
++
;
lapbeth
->
stats
.
rx_packets
++
;
...
@@ -150,16 +160,22 @@ static int lapbeth_rcv(struct sk_buff *skb, struct net_device *dev, struct packe
...
@@ -150,16 +160,22 @@ static int lapbeth_rcv(struct sk_buff *skb, struct net_device *dev, struct packe
skb_trim
(
skb
,
len
);
/* Set the length of the data */
skb_trim
(
skb
,
len
);
/* Set the length of the data */
if
((
err
=
lapb_data_received
(
lapbeth
,
skb
))
!=
LAPB_OK
)
{
if
((
err
=
lapb_data_received
(
lapbeth
,
skb
))
!=
LAPB_OK
)
{
kfree_skb
(
skb
);
printk
(
KERN_DEBUG
"lapbether: lapb_data_received err - %d
\n
"
,
err
);
printk
(
KERN_DEBUG
"lapbether: lapb_data_received err - %d
\n
"
,
err
);
goto
put_drop
;
}
}
lapbeth_put
(
lapbeth
);
out:
return
0
;
return
0
;
put_drop:
lapbeth_put
(
lapbeth
);
drop:
kfree_skb
(
skb
);
goto
out
;
}
}
static
int
lapbeth_data_indication
(
void
*
token
,
struct
sk_buff
*
skb
)
static
int
lapbeth_data_indication
(
void
*
token
,
struct
sk_buff
*
skb
)
{
{
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
token
;
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
token
;
unsigned
char
*
ptr
;
unsigned
char
*
ptr
;
ptr
=
skb_push
(
skb
,
1
);
ptr
=
skb_push
(
skb
,
1
);
...
@@ -178,8 +194,8 @@ static int lapbeth_data_indication(void *token, struct sk_buff *skb)
...
@@ -178,8 +194,8 @@ static int lapbeth_data_indication(void *token, struct sk_buff *skb)
*/
*/
static
int
lapbeth_xmit
(
struct
sk_buff
*
skb
,
struct
net_device
*
dev
)
static
int
lapbeth_xmit
(
struct
sk_buff
*
skb
,
struct
net_device
*
dev
)
{
{
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
int
err
;
int
err
=
-
ENODEV
;
/*
/*
* Just to be *really* sure not to send anything if the interface
* Just to be *really* sure not to send anything if the interface
...
@@ -187,50 +203,53 @@ static int lapbeth_xmit(struct sk_buff *skb, struct net_device *dev)
...
@@ -187,50 +203,53 @@ static int lapbeth_xmit(struct sk_buff *skb, struct net_device *dev)
*/
*/
if
(
!
netif_running
(
dev
))
{
if
(
!
netif_running
(
dev
))
{
lapbeth_check_devices
(
dev
);
lapbeth_check_devices
(
dev
);
kfree_skb
(
skb
);
goto
drop
;
return
-
ENODEV
;
}
}
switch
(
skb
->
data
[
0
])
{
switch
(
skb
->
data
[
0
])
{
case
0x00
:
case
0x00
:
break
;
err
=
0
;
case
0x01
:
break
;
if
((
err
=
lapb_connect_request
(
lapbeth
))
!=
LAPB_OK
)
case
0x01
:
printk
(
KERN_ERR
"lapbeth: lapb_connect_request error - %d
\n
"
,
err
);
if
((
err
=
lapb_connect_request
(
lapbeth
))
!=
LAPB_OK
)
kfree_skb
(
skb
);
printk
(
KERN_ERR
"lapbeth: lapb_connect_request "
return
0
;
"error: %d
\n
"
,
err
)
;
case
0x02
:
goto
drop_ok
;
if
((
err
=
lapb_disconnect_request
(
lapbeth
))
!=
LAPB_OK
)
case
0x02
:
printk
(
KERN_ERR
"lapbeth: lapb_disconnect_request err - %d
\n
"
,
err
);
if
((
err
=
lapb_disconnect_request
(
lapbeth
))
!=
LAPB_OK
)
kfree_skb
(
skb
);
printk
(
KERN_ERR
"lapbeth: lapb_disconnect_request "
return
0
;
"err: %d
\n
"
,
err
)
;
default:
/* Fall thru */
kfree_skb
(
skb
);
default:
return
0
;
goto
drop_ok
;
}
}
skb_pull
(
skb
,
1
);
skb_pull
(
skb
,
1
);
if
((
err
=
lapb_data_request
(
lapbeth
,
skb
))
!=
LAPB_OK
)
{
if
((
err
=
lapb_data_request
(
lapbeth
,
skb
))
!=
LAPB_OK
)
{
printk
(
KERN_ERR
"lapbeth: lapb_data_request error - %d
\n
"
,
err
);
printk
(
KERN_ERR
"lapbeth: lapb_data_request error - %d
\n
"
,
err
);
kfree_skb
(
skb
)
;
err
=
-
ENOMEM
;
return
-
ENOMEM
;
goto
drop
;
}
}
err
=
0
;
return
0
;
out:
return
err
;
drop_ok:
err
=
0
;
drop:
kfree_skb
(
skb
);
goto
out
;
}
}
static
void
lapbeth_data_transmit
(
void
*
token
,
struct
sk_buff
*
skb
)
static
void
lapbeth_data_transmit
(
void
*
token
,
struct
sk_buff
*
skb
)
{
{
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
token
;
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
token
;
unsigned
char
*
ptr
;
unsigned
char
*
ptr
;
struct
net_device
*
dev
;
struct
net_device
*
dev
;
int
size
;
int
size
=
skb
->
len
;
skb
->
protocol
=
htons
(
ETH_P_X25
);
skb
->
protocol
=
htons
(
ETH_P_X25
);
size
=
skb
->
len
;
ptr
=
skb_push
(
skb
,
2
);
ptr
=
skb_push
(
skb
,
2
);
*
ptr
++
=
size
%
256
;
*
ptr
++
=
size
%
256
;
...
@@ -247,11 +266,11 @@ static void lapbeth_data_transmit(void *token, struct sk_buff *skb)
...
@@ -247,11 +266,11 @@ static void lapbeth_data_transmit(void *token, struct sk_buff *skb)
static
void
lapbeth_connected
(
void
*
token
,
int
reason
)
static
void
lapbeth_connected
(
void
*
token
,
int
reason
)
{
{
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
token
;
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
token
;
struct
sk_buff
*
skb
;
unsigned
char
*
ptr
;
unsigned
char
*
ptr
;
struct
sk_buff
*
skb
=
dev_alloc_skb
(
1
);
if
(
(
skb
=
dev_alloc_skb
(
1
))
==
NULL
)
{
if
(
!
skb
)
{
printk
(
KERN_ERR
"lapbeth: out of memory
\n
"
);
printk
(
KERN_ERR
"lapbeth: out of memory
\n
"
);
return
;
return
;
}
}
...
@@ -269,11 +288,11 @@ static void lapbeth_connected(void *token, int reason)
...
@@ -269,11 +288,11 @@ static void lapbeth_connected(void *token, int reason)
static
void
lapbeth_disconnected
(
void
*
token
,
int
reason
)
static
void
lapbeth_disconnected
(
void
*
token
,
int
reason
)
{
{
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
token
;
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
token
;
struct
sk_buff
*
skb
;
unsigned
char
*
ptr
;
unsigned
char
*
ptr
;
struct
sk_buff
*
skb
=
dev_alloc_skb
(
1
);
if
(
(
skb
=
dev_alloc_skb
(
1
))
==
NULL
)
{
if
(
!
skb
)
{
printk
(
KERN_ERR
"lapbeth: out of memory
\n
"
);
printk
(
KERN_ERR
"lapbeth: out of memory
\n
"
);
return
;
return
;
}
}
...
@@ -294,7 +313,7 @@ static void lapbeth_disconnected(void *token, int reason)
...
@@ -294,7 +313,7 @@ static void lapbeth_disconnected(void *token, int reason)
*/
*/
static
struct
net_device_stats
*
lapbeth_get_stats
(
struct
net_device
*
dev
)
static
struct
net_device_stats
*
lapbeth_get_stats
(
struct
net_device
*
dev
)
{
{
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
return
&
lapbeth
->
stats
;
return
&
lapbeth
->
stats
;
}
}
...
@@ -303,9 +322,9 @@ static struct net_device_stats *lapbeth_get_stats(struct net_device *dev)
...
@@ -303,9 +322,9 @@ static struct net_device_stats *lapbeth_get_stats(struct net_device *dev)
*/
*/
static
int
lapbeth_set_mac_address
(
struct
net_device
*
dev
,
void
*
addr
)
static
int
lapbeth_set_mac_address
(
struct
net_device
*
dev
,
void
*
addr
)
{
{
struct
sockaddr
*
sa
=
(
struct
sockaddr
*
)
addr
;
struct
sockaddr
*
sa
=
(
struct
sockaddr
*
)
addr
;
memcpy
(
dev
->
dev_addr
,
sa
->
sa_data
,
dev
->
addr_len
);
memcpy
(
dev
->
dev_addr
,
sa
->
sa_data
,
dev
->
addr_len
);
return
0
;
return
0
;
}
}
/*
/*
...
@@ -320,7 +339,7 @@ static int lapbeth_open(struct net_device *dev)
...
@@ -320,7 +339,7 @@ static int lapbeth_open(struct net_device *dev)
if
(
lapbeth_check_devices
(
dev
))
if
(
lapbeth_check_devices
(
dev
))
return
-
ENODEV
;
/* oops, it's gone */
return
-
ENODEV
;
/* oops, it's gone */
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
lapbeth_callbacks
.
connect_confirmation
=
lapbeth_connected
;
lapbeth_callbacks
.
connect_confirmation
=
lapbeth_connected
;
lapbeth_callbacks
.
connect_indication
=
lapbeth_connected
;
lapbeth_callbacks
.
connect_indication
=
lapbeth_connected
;
...
@@ -340,7 +359,7 @@ static int lapbeth_open(struct net_device *dev)
...
@@ -340,7 +359,7 @@ static int lapbeth_open(struct net_device *dev)
static
int
lapbeth_close
(
struct
net_device
*
dev
)
static
int
lapbeth_close
(
struct
net_device
*
dev
)
{
{
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
struct
lapbethdev
*
lapbeth
=
(
struct
lapbethdev
*
)
dev
->
priv
;
int
err
;
int
err
;
netif_stop_queue
(
dev
);
netif_stop_queue
(
dev
);
...
@@ -358,20 +377,21 @@ static int lapbeth_close(struct net_device *dev)
...
@@ -358,20 +377,21 @@ static int lapbeth_close(struct net_device *dev)
*/
*/
static
int
lapbeth_new_device
(
struct
net_device
*
dev
)
static
int
lapbeth_new_device
(
struct
net_device
*
dev
)
{
{
int
k
;
unsigned
char
buf
[
14
];
unsigned
char
buf
[
14
];
struct
lapbethdev
*
lapbeth
,
*
lapbeth2
;
struct
lapbethdev
*
lapbeth
;
int
k
,
rc
=
-
ENOMEM
;
if
((
lapbeth
=
kmalloc
(
sizeof
(
struct
lapbethdev
),
GFP_
KERNEL
))
==
NULL
)
if
((
lapbeth
=
kmalloc
(
sizeof
(
struct
lapbethdev
),
GFP_
ATOMIC
))
==
NULL
)
return
-
ENOMEM
;
goto
out
;
memset
(
lapbeth
,
0
,
sizeof
(
struct
lapbethdev
));
memset
(
lapbeth
,
0
,
sizeof
(
struct
lapbethdev
));
dev_hold
(
dev
);
dev_hold
(
dev
);
lapbeth
->
ethdev
=
dev
;
lapbeth
->
ethdev
=
dev
;
lapbeth
->
ethname
[
sizeof
(
lapbeth
->
ethname
)
-
1
]
=
'\0'
;
strncpy
(
lapbeth
->
ethname
,
dev
->
name
,
sizeof
(
lapbeth
->
ethname
)
-
1
);
strncpy
(
lapbeth
->
ethname
,
dev
->
name
,
sizeof
(
lapbeth
->
ethname
)
-
1
);
lapbeth
->
ethname
[
sizeof
(
lapbeth
->
ethname
)
-
1
]
=
'\0'
;
atomic_set
(
&
lapbeth
->
refcnt
,
1
);
dev
=
&
lapbeth
->
axdev
;
dev
=
&
lapbeth
->
axdev
;
SET_MODULE_OWNER
(
dev
);
SET_MODULE_OWNER
(
dev
);
...
@@ -381,24 +401,21 @@ static int lapbeth_new_device(struct net_device *dev)
...
@@ -381,24 +401,21 @@ static int lapbeth_new_device(struct net_device *dev)
sprintf
(
buf
,
"lapb%d"
,
k
);
sprintf
(
buf
,
"lapb%d"
,
k
);
if
((
odev
=
__dev_get_by_name
(
buf
))
==
NULL
||
lapbeth_check_devices
(
odev
))
if
((
odev
=
__dev_get_by_name
(
buf
))
==
NULL
||
lapbeth_check_devices
(
odev
))
break
;
break
;
}
}
if
(
k
==
MAXLAPBDEV
)
{
rc
=
-
ENODEV
;
dev_put
(
dev
);
if
(
k
==
MAXLAPBDEV
)
kfree
(
lapbeth
);
goto
fail
;
return
-
ENODEV
;
}
dev
->
priv
=
(
void
*
)
lapbeth
;
/* pointer back */
dev
->
priv
=
(
void
*
)
lapbeth
;
/* pointer back */
strcpy
(
dev
->
name
,
buf
);
strcpy
(
dev
->
name
,
buf
);
if
(
register_netdev
(
dev
)
!=
0
)
{
rc
=
-
EIO
;
dev_put
(
dev
);
if
(
register_netdev
(
dev
))
kfree
(
lapbeth
);
goto
fail
;
return
-
EIO
;
}
dev
->
hard_start_xmit
=
lapbeth_xmit
;
dev
->
hard_start_xmit
=
lapbeth_xmit
;
dev
->
open
=
lapbeth_open
;
dev
->
open
=
lapbeth_open
;
...
@@ -410,26 +427,27 @@ static int lapbeth_new_device(struct net_device *dev)
...
@@ -410,26 +427,27 @@ static int lapbeth_new_device(struct net_device *dev)
dev
->
mtu
=
1000
;
dev
->
mtu
=
1000
;
dev
->
addr_len
=
0
;
dev
->
addr_len
=
0
;
cli
();
write_lock
(
&
lapbeth_devices_lock
);
list_add
(
&
lapbeth
->
node
,
&
lapbeth_devices
);
if
(
lapbeth_devices
==
NULL
)
{
lapbeth_hold
(
lapbeth
);
lapbeth_devices
=
lapbeth
;
write_unlock
(
&
lapbeth_devices_lock
);
}
else
{
rc
=
0
;
for
(
lapbeth2
=
lapbeth_devices
;
lapbeth2
->
next
!=
NULL
;
lapbeth2
=
lapbeth2
->
next
);
out:
lapbeth2
->
next
=
lapbeth
;
return
rc
;
}
fail:
dev_put
(
dev
);
sti
();
kfree
(
lapbeth
);
goto
out
;
return
0
;
}
}
/*
/*
* Handle device status changes.
* Handle device status changes.
*/
*/
static
int
lapbeth_device_event
(
struct
notifier_block
*
this
,
unsigned
long
event
,
void
*
ptr
)
static
int
lapbeth_device_event
(
struct
notifier_block
*
this
,
unsigned
long
event
,
void
*
ptr
)
{
{
struct
net_device
*
dev
=
(
struct
net_device
*
)
ptr
;
struct
lapbethdev
*
lapbeth
;
struct
net_device
*
dev
=
(
struct
net_device
*
)
ptr
;
if
(
!
dev_is_ethdev
(
dev
))
if
(
!
dev_is_ethdev
(
dev
))
return
NOTIFY_DONE
;
return
NOTIFY_DONE
;
...
@@ -437,19 +455,26 @@ static int lapbeth_device_event(struct notifier_block *this, unsigned long event
...
@@ -437,19 +455,26 @@ static int lapbeth_device_event(struct notifier_block *this, unsigned long event
lapbeth_check_devices
(
NULL
);
lapbeth_check_devices
(
NULL
);
switch
(
event
)
{
switch
(
event
)
{
case
NETDEV_UP
:
/* new ethernet device -> new LAPB interface */
case
NETDEV_UP
:
if
(
lapbeth_get_x25_dev
(
dev
)
==
NULL
)
/*
lapbeth_new_device
(
dev
);
* New ethernet device -> new LAPB interface
break
;
*/
lapbeth
=
lapbeth_get_x25_dev
(
dev
);
case
NETDEV_GOING_DOWN
:
case
NETDEV_DOWN
:
/* ethernet device closed -> close LAPB interface */
if
(
lapbeth
)
if
((
dev
=
lapbeth_get_x25_dev
(
dev
))
!=
NULL
)
lapbeth_put
(
lapbeth
);
dev_close
(
dev
);
else
break
;
lapbeth_new_device
(
dev
);
break
;
default:
case
NETDEV_GOING_DOWN
:
break
;
case
NETDEV_DOWN
:
/* ethernet device closed -> close LAPB interface */
lapbeth
=
lapbeth_get_x25_dev
(
dev
);
if
(
lapbeth
)
{
dev_close
(
lapbeth
->
ethdev
);
lapbeth_put
(
lapbeth
);
}
break
;
}
}
return
NOTIFY_DONE
;
return
NOTIFY_DONE
;
...
@@ -458,15 +483,15 @@ static int lapbeth_device_event(struct notifier_block *this, unsigned long event
...
@@ -458,15 +483,15 @@ static int lapbeth_device_event(struct notifier_block *this, unsigned long event
/* ------------------------------------------------------------------------ */
/* ------------------------------------------------------------------------ */
static
struct
packet_type
lapbeth_packet_type
=
{
static
struct
packet_type
lapbeth_packet_type
=
{
.
type
=
__constant_htons
(
ETH_P_DEC
),
.
type
=
__constant_htons
(
ETH_P_DEC
),
.
func
=
lapbeth_rcv
,
.
func
=
lapbeth_rcv
,
};
};
static
struct
notifier_block
lapbeth_dev_notifier
=
{
static
struct
notifier_block
lapbeth_dev_notifier
=
{
.
notifier_call
=
lapbeth_device_event
,
.
notifier_call
=
lapbeth_device_event
,
};
};
static
char
banner
[]
__initdata
=
KERN_INFO
"LAPB Ethernet driver version 0.0
1
\n
"
;
static
char
banner
[]
__initdata
=
KERN_INFO
"LAPB Ethernet driver version 0.0
2
\n
"
;
static
int
__init
lapbeth_init_driver
(
void
)
static
int
__init
lapbeth_init_driver
(
void
)
{
{
...
@@ -479,7 +504,7 @@ static int __init lapbeth_init_driver(void)
...
@@ -479,7 +504,7 @@ static int __init lapbeth_init_driver(void)
printk
(
banner
);
printk
(
banner
);
read_lock_bh
(
&
dev_base_lock
);
read_lock_bh
(
&
dev_base_lock
);
for
(
dev
=
dev_base
;
dev
!=
NULL
;
dev
=
dev
->
next
)
{
for
(
dev
=
dev_base
;
dev
;
dev
=
dev
->
next
)
{
if
(
dev_is_ethdev
(
dev
))
{
if
(
dev_is_ethdev
(
dev
))
{
read_unlock_bh
(
&
dev_base_lock
);
read_unlock_bh
(
&
dev_base_lock
);
lapbeth_new_device
(
dev
);
lapbeth_new_device
(
dev
);
...
@@ -495,18 +520,24 @@ module_init(lapbeth_init_driver);
...
@@ -495,18 +520,24 @@ module_init(lapbeth_init_driver);
static
void
__exit
lapbeth_cleanup_driver
(
void
)
static
void
__exit
lapbeth_cleanup_driver
(
void
)
{
{
struct
lapbethdev
*
lapbeth
;
struct
lapbethdev
*
lapbeth
;
struct
list_head
*
entry
,
*
tmp
;
dev_remove_pack
(
&
lapbeth_packet_type
);
dev_remove_pack
(
&
lapbeth_packet_type
);
unregister_netdevice_notifier
(
&
lapbeth_dev_notifier
);
unregister_netdevice_notifier
(
&
lapbeth_dev_notifier
);
for
(
lapbeth
=
lapbeth_devices
;
lapbeth
!=
NULL
;
lapbeth
=
lapbeth
->
next
)
write_lock
(
&
lapbeth_devices_lock
);
list_for_each_safe
(
entry
,
tmp
,
&
lapbeth_devices
)
{
lapbeth
=
list_entry
(
entry
,
struct
lapbethdev
,
node
);
unregister_netdev
(
&
lapbeth
->
axdev
);
unregister_netdev
(
&
lapbeth
->
axdev
);
list_del
(
&
lapbeth
->
node
);
lapbeth_put
(
lapbeth
);
}
write_unlock
(
&
lapbeth_devices_lock
);
}
}
module_exit
(
lapbeth_cleanup_driver
);
module_exit
(
lapbeth_cleanup_driver
);
MODULE_AUTHOR
(
"Jonathan Naylor <g4klx@g4klx.demon.co.uk>"
);
MODULE_AUTHOR
(
"Jonathan Naylor <g4klx@g4klx.demon.co.uk>"
);
MODULE_DESCRIPTION
(
"The unofficial LAPB over Ethernet driver"
);
MODULE_DESCRIPTION
(
"The unofficial LAPB over Ethernet driver"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_LICENSE
(
"GPL"
);
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment