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
Kirill Smelkov
linux
Commits
d911e854
Commit
d911e854
authored
Dec 28, 2003
by
Ben Collins
Browse files
Options
Browse Files
Download
Plain Diff
Merge
http://linux.bkbits.net/linux-2.5
into debian.org:/usr/src/kernel/linux-2.6
parents
fe6705c3
af17e3f3
Changes
19
Hide whitespace changes
Inline
Side-by-side
Showing
19 changed files
with
470 additions
and
423 deletions
+470
-423
drivers/ieee1394/Kconfig
drivers/ieee1394/Kconfig
+10
-10
drivers/ieee1394/eth1394.c
drivers/ieee1394/eth1394.c
+8
-8
drivers/ieee1394/highlevel.c
drivers/ieee1394/highlevel.c
+24
-20
drivers/ieee1394/highlevel.h
drivers/ieee1394/highlevel.h
+3
-2
drivers/ieee1394/hosts.c
drivers/ieee1394/hosts.c
+21
-83
drivers/ieee1394/hosts.h
drivers/ieee1394/hosts.h
+3
-19
drivers/ieee1394/ieee1394_core.c
drivers/ieee1394/ieee1394_core.c
+63
-57
drivers/ieee1394/ieee1394_core.h
drivers/ieee1394/ieee1394_core.h
+12
-12
drivers/ieee1394/ieee1394_transactions.c
drivers/ieee1394/ieee1394_transactions.c
+29
-40
drivers/ieee1394/iso.c
drivers/ieee1394/iso.c
+12
-4
drivers/ieee1394/iso.h
drivers/ieee1394/iso.h
+11
-0
drivers/ieee1394/nodemgr.c
drivers/ieee1394/nodemgr.c
+46
-42
drivers/ieee1394/nodemgr.h
drivers/ieee1394/nodemgr.h
+9
-0
drivers/ieee1394/ohci1394.c
drivers/ieee1394/ohci1394.c
+31
-20
drivers/ieee1394/oui.db
drivers/ieee1394/oui.db
+3
-3
drivers/ieee1394/pcilynx.c
drivers/ieee1394/pcilynx.c
+17
-13
drivers/ieee1394/raw1394.c
drivers/ieee1394/raw1394.c
+116
-10
drivers/ieee1394/raw1394.h
drivers/ieee1394/raw1394.h
+5
-0
drivers/ieee1394/sbp2.c
drivers/ieee1394/sbp2.c
+47
-80
No files found.
drivers/ieee1394/Kconfig
View file @
d911e854
# -*- shell-script -*-
menu "IEEE 1394 (FireWire) support (EXPERIMENTAL)"
depends on
PCI &&
EXPERIMENTAL
depends on EXPERIMENTAL
config IEEE1394
tristate "IEEE 1394 (FireWire) support (EXPERIMENTAL)"
...
...
@@ -15,7 +15,7 @@ config IEEE1394
is the core support only, you will also need to select a driver for
your IEEE 1394 adapter.
To compile this driver as a module,
choose
M here: the
To compile this driver as a module,
say
M here: the
module will be called ieee1394.
comment "Subsystem Options"
...
...
@@ -56,13 +56,13 @@ comment "Texas Instruments PCILynx requires I2C bit-banging"
config IEEE1394_PCILYNX
tristate "Texas Instruments PCILynx support"
depends on IEEE1394 && I2C_ALGOBIT
depends on
PCI &&
IEEE1394 && I2C_ALGOBIT
help
Say Y here if you have an IEEE-1394 controller with the Texas
Instruments PCILynx chip. Note: this driver is written for revision
2 of this chip and may not work with revision 0.
To compile this driver as a module,
choose
M here: the
To compile this driver as a module,
say
M here: the
module will be called pcilynx.
# Non-maintained pcilynx options
...
...
@@ -72,7 +72,7 @@ config IEEE1394_PCILYNX
# fi
config IEEE1394_OHCI1394
tristate "OHCI-1394 support"
depends on IEEE1394
depends on
PCI &&
IEEE1394
help
Enable this driver if you have an IEEE 1394 controller based on the
OHCI-1394 specification. The current driver is only tested with OHCI
...
...
@@ -80,7 +80,7 @@ config IEEE1394_OHCI1394
use one of these chipsets. It should work with any OHCI-1394
compliant card, however.
To compile this driver as a module,
choose
M here: the
To compile this driver as a module,
say
M here: the
module will be called ohci1394.
comment "Protocol Drivers"
...
...
@@ -122,7 +122,7 @@ config IEEE1394_DV1394
The user-space API for dv1394 is documented in dv1394.h.
To compile this driver as a module,
choose
M here: the
To compile this driver as a module,
say
M here: the
module will be called dv1394.
config IEEE1394_RAWIO
...
...
@@ -134,7 +134,7 @@ config IEEE1394_RAWIO
direct communication of user programs with the IEEE 1394 bus and
thus with the attached peripherals.
To compile this driver as a module,
choose
M here: the
To compile this driver as a module,
say
M here: the
module will be called raw1394.
config IEEE1394_CMP
...
...
@@ -144,7 +144,7 @@ config IEEE1394_CMP
This option enables the Connection Management Procedures
(IEC61883-1) driver, which implements input and output plugs.
To compile this driver as a module,
choose
M here: the
To compile this driver as a module,
say
M here: the
module will be called cmp.
config IEEE1394_AMDTP
...
...
@@ -157,7 +157,7 @@ config IEEE1394_AMDTP
The userspace interface is documented in amdtp.h.
To compile this driver as a module,
choose
M here: the
To compile this driver as a module,
say
M here: the
module will be called amdtp.
endmenu
drivers/ieee1394/eth1394.c
View file @
d911e854
...
...
@@ -89,7 +89,7 @@
#define TRACE() printk(KERN_ERR "%s:%s[%d] ---- TRACE\n", driver_name, __FUNCTION__, __LINE__)
static
char
version
[]
__devinitdata
=
"$Rev: 10
43
$ Ben Collins <bcollins@debian.org>"
;
"$Rev: 10
79
$ Ben Collins <bcollins@debian.org>"
;
struct
fragment_info
{
struct
list_head
list
;
...
...
@@ -220,7 +220,7 @@ static int ether1394_init_bc(struct net_device *dev)
priv
->
iso
=
hpsb_iso_recv_init
(
priv
->
host
,
16
*
4096
,
16
,
priv
->
broadcast_channel
,
1
,
ether1394_iso
);
HPSB_ISO_DMA_PACKET_PER_BUFFER
,
1
,
ether1394_iso
);
if
(
priv
->
iso
==
NULL
)
{
ETH1394_PRINT
(
KERN_ERR
,
dev
->
name
,
"failed to change broadcast "
...
...
@@ -475,7 +475,7 @@ static void ether1394_add_host (struct hpsb_host *host)
priv
->
broadcast_channel
=
host
->
csr
.
broadcast_channel
&
0x3f
;
priv
->
iso
=
hpsb_iso_recv_init
(
host
,
16
*
4096
,
16
,
priv
->
broadcast_channel
,
1
,
ether1394_iso
);
HPSB_ISO_DMA_PACKET_PER_BUFFER
,
1
,
ether1394_iso
);
if
(
priv
->
iso
==
NULL
)
{
priv
->
bc_state
=
ETHER1394_BC_CLOSED
;
}
...
...
@@ -1258,7 +1258,7 @@ static inline struct hpsb_packet *ether1394_alloc_common_packet(struct hpsb_host
{
struct
hpsb_packet
*
p
;
p
=
alloc_hpsb
_packet
(
0
);
p
=
hpsb_alloc
_packet
(
0
);
if
(
p
)
{
p
->
host
=
host
;
p
->
data
=
NULL
;
...
...
@@ -1327,7 +1327,7 @@ static inline void ether1394_free_packet(struct hpsb_packet *packet)
if
(
packet
->
tcode
!=
TCODE_STREAM_DATA
)
hpsb_free_tlabel
(
packet
);
packet
->
data
=
NULL
;
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
}
static
void
ether1394_complete_cb
(
void
*
__ptask
);
...
...
@@ -1349,7 +1349,7 @@ static int ether1394_send_packet(struct packet_task *ptask, unsigned int tx_len)
ptask
->
dest_node
,
ptask
->
addr
,
ptask
->
skb
->
data
,
tx_len
))
{
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
-
1
;
}
...
...
@@ -1357,7 +1357,7 @@ static int ether1394_send_packet(struct packet_task *ptask, unsigned int tx_len)
hpsb_set_packet_complete_task
(
ptask
->
packet
,
ether1394_complete_cb
,
ptask
);
if
(
!
hpsb_send_packet
(
packet
)
)
{
if
(
hpsb_send_packet
(
packet
)
<
0
)
{
ether1394_free_packet
(
packet
);
return
-
1
;
}
...
...
@@ -1599,7 +1599,7 @@ static int ether1394_ethtool_ioctl(struct net_device *dev, void *useraddr)
case
ETHTOOL_GDRVINFO
:
{
struct
ethtool_drvinfo
info
=
{
ETHTOOL_GDRVINFO
};
strcpy
(
info
.
driver
,
driver_name
);
strcpy
(
info
.
version
,
"$Rev: 10
43
$"
);
strcpy
(
info
.
version
,
"$Rev: 10
79
$"
);
/* FIXME XXX provide sane businfo */
strcpy
(
info
.
bus_info
,
"ieee1394"
);
if
(
copy_to_user
(
useraddr
,
&
info
,
sizeof
(
info
)))
...
...
drivers/ieee1394/highlevel.c
View file @
d911e854
...
...
@@ -26,6 +26,7 @@
#include "hosts.h"
#include "ieee1394_core.h"
#include "highlevel.h"
#include "nodemgr.h"
struct
hl_host_info
{
...
...
@@ -227,10 +228,17 @@ struct hpsb_host *hpsb_get_host_bykey(struct hpsb_highlevel *hl, unsigned long k
return
host
;
}
static
int
highlevel_for_each_host_reg
(
struct
hpsb_host
*
host
,
void
*
__data
)
{
struct
hpsb_highlevel
*
hl
=
__data
;
hl
->
add_host
(
host
);
return
0
;
}
void
hpsb_register_highlevel
(
struct
hpsb_highlevel
*
hl
)
{
struct
list_head
*
lh
;
unsigned
long
flags
;
INIT_LIST_HEAD
(
&
hl
->
addr_list
);
...
...
@@ -242,21 +250,25 @@ void hpsb_register_highlevel(struct hpsb_highlevel *hl)
list_add_tail
(
&
hl
->
hl_list
,
&
hl_drivers
);
write_unlock_irqrestore
(
&
hl_drivers_lock
,
flags
);
if
(
hl
->
add_host
)
{
down
(
&
hpsb_hosts_lock
);
list_for_each
(
lh
,
&
hpsb_hosts
)
{
struct
hpsb_host
*
host
=
list_entry
(
lh
,
struct
hpsb_host
,
host_list
);
hl
->
add_host
(
host
);
}
up
(
&
hpsb_hosts_lock
);
}
if
(
hl
->
add_host
)
nodemgr_for_each_host
(
hl
,
highlevel_for_each_host_reg
);
return
;
}
static
int
highlevel_for_each_host_unreg
(
struct
hpsb_host
*
host
,
void
*
__data
)
{
struct
hpsb_highlevel
*
hl
=
__data
;
hl
->
remove_host
(
host
);
hpsb_destroy_hostinfo
(
hl
,
host
);
return
0
;
}
void
hpsb_unregister_highlevel
(
struct
hpsb_highlevel
*
hl
)
{
struct
list_head
*
lh
,
*
next
;
struct
list_head
*
lh
,
*
next
;
struct
hpsb_address_serve
*
as
;
unsigned
long
flags
;
...
...
@@ -272,16 +284,8 @@ void hpsb_unregister_highlevel(struct hpsb_highlevel *hl)
list_del
(
&
hl
->
hl_list
);
write_unlock_irqrestore
(
&
hl_drivers_lock
,
flags
);
if
(
hl
->
remove_host
)
{
down
(
&
hpsb_hosts_lock
);
list_for_each
(
lh
,
&
hpsb_hosts
)
{
struct
hpsb_host
*
host
=
list_entry
(
lh
,
struct
hpsb_host
,
host_list
);
hl
->
remove_host
(
host
);
hpsb_destroy_hostinfo
(
hl
,
host
);
}
up
(
&
hpsb_hosts_lock
);
}
if
(
hl
->
remove_host
)
nodemgr_for_each_host
(
hl
,
highlevel_for_each_host_unreg
);
}
int
hpsb_register_addrspace
(
struct
hpsb_highlevel
*
hl
,
...
...
drivers/ieee1394/highlevel.h
View file @
d911e854
...
...
@@ -92,8 +92,6 @@ struct hpsb_address_ops {
};
void
init_hpsb_highlevel
(
void
);
void
highlevel_add_host
(
struct
hpsb_host
*
host
);
void
highlevel_remove_host
(
struct
hpsb_host
*
host
);
void
highlevel_host_reset
(
struct
hpsb_host
*
host
);
...
...
@@ -180,4 +178,7 @@ int hpsb_set_hostinfo(struct hpsb_highlevel *hl, struct hpsb_host *host, void *d
/* Retrieve hpsb_host using a highlevel handle and a key */
struct
hpsb_host
*
hpsb_get_host_bykey
(
struct
hpsb_highlevel
*
hl
,
unsigned
long
key
);
/* Initialize the highlevel system */
void
init_hpsb_highlevel
(
void
);
#endif
/* IEEE1394_HIGHLEVEL_H */
drivers/ieee1394/hosts.c
View file @
d911e854
...
...
@@ -16,14 +16,14 @@
#include <linux/list.h>
#include <linux/init.h>
#include <linux/slab.h>
#include <linux/pci.h>
#include "ieee1394_types.h"
#include "hosts.h"
#include "ieee1394_core.h"
#include "highlevel.h"
#include "nodemgr.h"
LIST_HEAD
(
hpsb_hosts
);
DECLARE_MUTEX
(
hpsb_hosts_lock
);
static
int
dummy_transmit_packet
(
struct
hpsb_host
*
h
,
struct
hpsb_packet
*
p
)
{
...
...
@@ -46,54 +46,14 @@ static struct hpsb_host_driver dummy_driver = {
.
isoctl
=
dummy_isoctl
};
/**
* hpsb_ref_host - increase reference count for host controller.
* @host: the host controller
*
* Increase the reference count for the specified host controller.
* When holding a reference to a host, the memory allocated for the
* host struct will not be freed and the host is guaranteed to be in a
* consistent state. The driver may be unloaded or the controller may
* be removed (PCMCIA), but the host struct will remain valid.
*/
int
hpsb_ref_host
(
struct
hpsb_host
*
host
)
static
int
alloc_hostnum_cb
(
struct
hpsb_host
*
host
,
void
*
__data
)
{
struct
list_head
*
lh
;
int
retval
=
0
;
down
(
&
hpsb_hosts_lock
);
list_for_each
(
lh
,
&
hpsb_hosts
)
{
if
(
host
==
list_entry
(
lh
,
struct
hpsb_host
,
host_list
))
{
if
(
try_module_get
(
host
->
driver
->
owner
))
{
atomic_inc
(
&
host
->
refcount
);
retval
=
1
;
}
break
;
}
}
up
(
&
hpsb_hosts_lock
);
int
*
hostnum
=
__data
;
return
retval
;
}
if
(
host
->
id
==
*
hostnum
)
return
1
;
/**
* hpsb_unref_host - decrease reference count for host controller.
* @host: the host controller
*
* Decrease the reference count for the specified host controller.
* When the reference count reaches zero, the memory allocated for the
* &hpsb_host will be freed.
*/
void
hpsb_unref_host
(
struct
hpsb_host
*
host
)
{
module_put
(
host
->
driver
->
owner
);
down
(
&
hpsb_hosts_lock
);
if
(
atomic_dec_and_test
(
&
host
->
refcount
)
&&
host
->
is_shutdown
)
device_unregister
(
&
host
->
device
);
up
(
&
hpsb_hosts_lock
);
return
0
;
}
/**
...
...
@@ -108,19 +68,16 @@ void hpsb_unref_host(struct hpsb_host *host)
* driver specific parts, enable the controller and make it available
* to the general subsystem using hpsb_add_host().
*
* The &hpsb_host is allocated with an single initial reference
* belonging to the driver. Once the driver is done with the struct,
* for example, when the driver is unloaded, it should release this
* reference using hpsb_unref_host().
*
* Return Value: a pointer to the &hpsb_host if succesful, %NULL if
* no memory was available.
*/
struct
hpsb_host
*
hpsb_alloc_host
(
struct
hpsb_host_driver
*
drv
,
size_t
extra
)
struct
hpsb_host
*
hpsb_alloc_host
(
struct
hpsb_host_driver
*
drv
,
size_t
extra
,
struct
device
*
dev
)
{
struct
hpsb_host
*
h
;
int
i
;
int
hostnum
=
0
;
h
=
kmalloc
(
sizeof
(
struct
hpsb_host
)
+
extra
,
SLAB_KERNEL
);
if
(
!
h
)
return
NULL
;
...
...
@@ -128,7 +85,6 @@ struct hpsb_host *hpsb_alloc_host(struct hpsb_host_driver *drv, size_t extra)
h
->
hostdata
=
h
+
1
;
h
->
driver
=
drv
;
atomic_set
(
&
h
->
refcount
,
1
);
INIT_LIST_HEAD
(
&
h
->
pending_packets
);
spin_lock_init
(
&
h
->
pending_pkt_lock
);
...
...
@@ -146,53 +102,35 @@ struct hpsb_host *hpsb_alloc_host(struct hpsb_host_driver *drv, size_t extra)
h
->
topology_map
=
h
->
csr
.
topology_map
+
3
;
h
->
speed_map
=
(
u8
*
)(
h
->
csr
.
speed_map
+
2
);
return
h
;
}
static
int
alloc_hostnum
(
void
)
{
int
hostnum
=
0
;
while
(
1
)
{
struct
list_head
*
lh
;
int
found
=
0
;
list_for_each
(
lh
,
&
hpsb_hosts
)
{
struct
hpsb_host
*
host
=
list_entry
(
lh
,
struct
hpsb_host
,
host_list
);
if
(
host
->
id
==
hostnum
)
{
found
=
1
;
break
;
}
if
(
!
nodemgr_for_each_host
(
&
hostnum
,
alloc_hostnum_cb
))
{
h
->
id
=
hostnum
;
break
;
}
if
(
!
found
)
return
hostnum
;
hostnum
++
;
}
return
0
;
memcpy
(
&
h
->
device
,
&
nodemgr_dev_template_host
,
sizeof
(
h
->
device
));
h
->
device
.
parent
=
dev
;
snprintf
(
h
->
device
.
bus_id
,
BUS_ID_SIZE
,
"fw-host%d"
,
h
->
id
);
device_register
(
&
h
->
device
);
return
h
;
}
void
hpsb_add_host
(
struct
hpsb_host
*
host
)
{
down
(
&
hpsb_hosts_lock
);
host
->
id
=
alloc_hostnum
();
list_add_tail
(
&
host
->
host_list
,
&
hpsb_hosts
);
up
(
&
hpsb_hosts_lock
);
highlevel_add_host
(
host
);
host
->
driver
->
devctl
(
host
,
RESET_BUS
,
LONG_RESET
);
}
void
hpsb_remove_host
(
struct
hpsb_host
*
host
)
{
down
(
&
hpsb_hosts_lock
);
host
->
is_shutdown
=
1
;
host
->
driver
=
&
dummy_driver
;
list_del
(
&
host
->
host_list
);
up
(
&
hpsb_hosts_lock
);
highlevel_remove_host
(
host
);
device_unregister
(
&
host
->
device
);
}
drivers/ieee1394/hosts.h
View file @
d911e854
...
...
@@ -29,8 +29,6 @@ struct hpsb_host {
atomic_t
generation
;
atomic_t
refcount
;
struct
list_head
pending_packets
;
spinlock_t
pending_pkt_lock
;
struct
timer_list
timeout
;
...
...
@@ -165,7 +163,7 @@ struct hpsb_host_driver {
* packet->type == raw) and do byte-swapping as necessary or instruct
* the hardware to do so. It can return immediately after the packet
* was queued for sending. After sending, hpsb_sent_packet() has to be
* called. Return 0
for
failure.
* called. Return 0
on success, negative errno on
failure.
* NOTE: The function must be callable in interrupt context.
*/
int
(
*
transmit_packet
)
(
struct
hpsb_host
*
host
,
...
...
@@ -195,22 +193,8 @@ struct hpsb_host_driver {
};
extern
struct
list_head
hpsb_hosts
;
extern
struct
semaphore
hpsb_hosts_lock
;
/*
* In order to prevent hosts from unloading, use hpsb_ref_host(). This prevents
* the host from going away (e.g. makes module unloading of the driver
* impossible), but still can not guarantee it (e.g. PC-Card being pulled by the
* user). hpsb_ref_host() returns false if host could not be locked. If it is
* successful, host is valid as a pointer until hpsb_unref_host() (not just
* until after remove_host).
*/
int
hpsb_ref_host
(
struct
hpsb_host
*
host
);
void
hpsb_unref_host
(
struct
hpsb_host
*
host
);
struct
hpsb_host
*
hpsb_alloc_host
(
struct
hpsb_host_driver
*
drv
,
size_t
extra
);
struct
hpsb_host
*
hpsb_alloc_host
(
struct
hpsb_host_driver
*
drv
,
size_t
extra
,
struct
device
*
dev
);
void
hpsb_add_host
(
struct
hpsb_host
*
host
);
void
hpsb_remove_host
(
struct
hpsb_host
*
h
);
...
...
drivers/ieee1394/ieee1394_core.c
View file @
d911e854
...
...
@@ -109,7 +109,7 @@ void hpsb_set_packet_complete_task(struct hpsb_packet *packet,
}
/**
*
alloc_hpsb
_packet - allocate new packet structure
*
hpsb_alloc
_packet - allocate new packet structure
* @data_size: size of the data block to be allocated
*
* This function allocates, initializes and returns a new &struct hpsb_packet.
...
...
@@ -128,7 +128,7 @@ void hpsb_set_packet_complete_task(struct hpsb_packet *packet,
* Return value: A pointer to a &struct hpsb_packet or NULL on allocation
* failure.
*/
struct
hpsb_packet
*
alloc_hpsb
_packet
(
size_t
data_size
)
struct
hpsb_packet
*
hpsb_alloc
_packet
(
size_t
data_size
)
{
struct
hpsb_packet
*
packet
=
NULL
;
void
*
data
=
NULL
;
...
...
@@ -152,25 +152,23 @@ struct hpsb_packet *alloc_hpsb_packet(size_t data_size)
}
INIT_LIST_HEAD
(
&
packet
->
list
);
sema_init
(
&
packet
->
state_change
,
0
);
packet
->
complete_routine
=
NULL
;
packet
->
complete_data
=
NULL
;
packet
->
state
=
hpsb_unused
;
packet
->
generation
=
-
1
;
packet
->
data_be
=
1
;
return
packet
;
}
/**
*
free_hpsb
_packet - free packet and data associated with it
*
hpsb_free
_packet - free packet and data associated with it
* @packet: packet to free (is NULL safe)
*
* This function will free packet->data, packet->header and finally the packet
* itself.
*/
void
free_hpsb
_packet
(
struct
hpsb_packet
*
packet
)
void
hpsb_free
_packet
(
struct
hpsb_packet
*
packet
)
{
if
(
!
packet
)
return
;
...
...
@@ -420,14 +418,12 @@ void hpsb_packet_sent(struct hpsb_host *host, struct hpsb_packet *packet,
if
(
packet
->
no_waiter
)
{
/* must not have a tlabel allocated */
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
;
}
if
(
ackcode
!=
ACK_PENDING
||
!
packet
->
expect_response
)
{
packet
->
state
=
hpsb_complete
;
up
(
&
packet
->
state_change
);
up
(
&
packet
->
state_change
);
run_packet_complete
(
packet
);
return
;
}
...
...
@@ -439,7 +435,6 @@ void hpsb_packet_sent(struct hpsb_host *host, struct hpsb_packet *packet,
list_add_tail
(
&
packet
->
list
,
&
host
->
pending_packets
);
spin_unlock_irqrestore
(
&
host
->
pending_pkt_lock
,
flags
);
up
(
&
packet
->
state_change
);
mod_timer
(
&
host
->
timeout
,
jiffies
+
host
->
timeout_interval
);
}
...
...
@@ -465,7 +460,7 @@ int hpsb_send_phy_config(struct hpsb_host *host, int rootid, int gapcnt)
return
-
EINVAL
;
}
packet
=
alloc_hpsb
_packet
(
0
);
packet
=
hpsb_alloc
_packet
(
0
);
if
(
!
packet
)
return
-
ENOMEM
;
...
...
@@ -485,16 +480,8 @@ int hpsb_send_phy_config(struct hpsb_host *host, int rootid, int gapcnt)
packet
->
generation
=
get_hpsb_generation
(
host
);
if
(
!
hpsb_send_packet
(
packet
))
{
retval
=
-
EINVAL
;
goto
fail
;
}
down
(
&
packet
->
state_change
);
down
(
&
packet
->
state_change
);
fail:
free_hpsb_packet
(
packet
);
retval
=
hpsb_send_packet_and_wait
(
packet
);
hpsb_free_packet
(
packet
);
return
retval
;
}
...
...
@@ -504,23 +491,24 @@ int hpsb_send_phy_config(struct hpsb_host *host, int rootid, int gapcnt)
* @packet: packet to send
*
* The packet is sent through the host specified in the packet->host field.
* Before sending, the packet's transmit speed is automatically determined
using
* the local speed map when it is an async, non-broadcast packet.
* Before sending, the packet's transmit speed is automatically determined
*
using
the local speed map when it is an async, non-broadcast packet.
*
* Possibilities for failure are that host is either not initialized, in bus
* reset, the packet's generation number doesn't match the current generation
* number or the host reports a transmit error.
*
* Return value:
False (0) on failure, true (1) otherwis
e.
* Return value:
0 on success, negative errno on failur
e.
*/
int
hpsb_send_packet
(
struct
hpsb_packet
*
packet
)
{
struct
hpsb_host
*
host
=
packet
->
host
;
if
(
host
->
is_shutdown
||
host
->
in_bus_reset
||
(
packet
->
generation
!=
get_hpsb_generation
(
host
)))
{
return
0
;
}
if
(
host
->
is_shutdown
)
return
-
EINVAL
;
if
(
host
->
in_bus_reset
||
(
packet
->
generation
!=
get_hpsb_generation
(
host
)))
return
-
EAGAIN
;
packet
->
state
=
hpsb_queued
;
...
...
@@ -532,23 +520,13 @@ int hpsb_send_packet(struct hpsb_packet *packet)
data
=
kmalloc
(
packet
->
header_size
+
packet
->
data_size
,
GFP_ATOMIC
);
if
(
!
data
)
{
HPSB_ERR
(
"unable to allocate memory for concatenating header and data"
);
return
0
;
return
-
ENOMEM
;
}
memcpy
(
data
,
packet
->
header
,
packet
->
header_size
);
if
(
packet
->
data_size
)
{
if
(
packet
->
data_be
)
{
memcpy
(((
u8
*
)
data
)
+
packet
->
header_size
,
packet
->
data
,
packet
->
data_size
);
}
else
{
int
i
;
quadlet_t
*
my_data
=
(
quadlet_t
*
)
((
u8
*
)
data
+
packet
->
data_size
);
for
(
i
=
0
;
i
<
packet
->
data_size
/
4
;
i
++
)
{
my_data
[
i
]
=
cpu_to_be32
(
packet
->
data
[
i
]);
}
}
}
memcpy
(((
u8
*
)
data
)
+
packet
->
header_size
,
packet
->
data
,
packet
->
data_size
);
dump_packet
(
"send packet local:"
,
packet
->
header
,
packet
->
header_size
);
...
...
@@ -558,7 +536,7 @@ int hpsb_send_packet(struct hpsb_packet *packet)
kfree
(
data
);
return
1
;
return
0
;
}
if
(
packet
->
type
==
hpsb_async
&&
packet
->
node_id
!=
ALL_NODES
)
{
...
...
@@ -584,10 +562,33 @@ int hpsb_send_packet(struct hpsb_packet *packet)
return
host
->
driver
->
transmit_packet
(
host
,
packet
);
}
/* We could just use complete() directly as the packet complete
* callback, but this is more typesafe, in the sense that we get a
* compiler error if the prototype for complete() changes. */
static
void
complete_packet
(
void
*
data
)
{
complete
((
struct
completion
*
)
data
);
}
int
hpsb_send_packet_and_wait
(
struct
hpsb_packet
*
packet
)
{
struct
completion
done
;
int
retval
;
init_completion
(
&
done
);
hpsb_set_packet_complete_task
(
packet
,
complete_packet
,
&
done
);
retval
=
hpsb_send_packet
(
packet
);
if
(
retval
==
0
)
wait_for_completion
(
&
done
);
return
retval
;
}
static
void
send_packet_nocare
(
struct
hpsb_packet
*
packet
)
{
if
(
!
hpsb_send_packet
(
packet
)
)
{
free_hpsb
_packet
(
packet
);
if
(
hpsb_send_packet
(
packet
)
<
0
)
{
hpsb_free
_packet
(
packet
);
}
}
...
...
@@ -668,7 +669,6 @@ void handle_packet_response(struct hpsb_host *host, int tcode, quadlet_t *data,
}
packet
->
state
=
hpsb_complete
;
up
(
&
packet
->
state_change
);
run_packet_complete
(
packet
);
}
...
...
@@ -680,7 +680,7 @@ static struct hpsb_packet *create_reply_packet(struct hpsb_host *host,
dsize
+=
(
dsize
%
4
?
4
-
(
dsize
%
4
)
:
0
);
p
=
alloc_hpsb
_packet
(
dsize
);
p
=
hpsb_alloc
_packet
(
dsize
);
if
(
p
==
NULL
)
{
/* FIXME - send data_error response */
return
NULL
;
...
...
@@ -820,6 +820,8 @@ static void handle_incoming_packet(struct hpsb_host *host, int tcode,
if
(
rcode
>=
0
)
{
fill_async_readblock_resp
(
packet
,
rcode
,
length
);
send_packet_nocare
(
packet
);
}
else
{
hpsb_free_packet
(
packet
);
}
break
;
...
...
@@ -874,7 +876,7 @@ static void handle_incoming_packet(struct hpsb_host *host, int tcode,
if
(
rcode
>=
0
)
{
send_packet_nocare
(
packet
);
}
else
{
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
}
break
;
}
...
...
@@ -949,7 +951,6 @@ void abort_requests(struct hpsb_host *host)
list_del
(
&
packet
->
list
);
packet
->
state
=
hpsb_complete
;
packet
->
ack_code
=
ACKX_ABORTED
;
up
(
&
packet
->
state_change
);
run_packet_complete
(
packet
);
}
}
...
...
@@ -987,7 +988,6 @@ void abort_timedouts(unsigned long __opaque)
list_del
(
&
packet
->
list
);
packet
->
state
=
hpsb_complete
;
packet
->
ack_code
=
ACKX_TIMEOUT
;
up
(
&
packet
->
state_change
);
run_packet_complete
(
packet
);
}
}
...
...
@@ -1171,19 +1171,22 @@ static int ieee1394_dispatch_open(struct inode *inode, struct file *file)
static
int
__init
ieee1394_init
(
void
)
{
hpsb_packet_cache
=
kmem_cache_create
(
"hpsb_packet"
,
sizeof
(
struct
hpsb_packet
),
0
,
0
,
NULL
,
NULL
);
devfs_mk_dir
(
"ieee1394"
);
if
(
register_chrdev
(
IEEE1394_MAJOR
,
"ieee1394"
,
&
ieee1394_chardev_ops
))
{
HPSB_ERR
(
"unable to register character device major %d!
\n
"
,
IEEE1394_MAJOR
);
devfs_remove
(
"ieee1394"
);
return
-
ENODEV
;
}
devfs_mk_dir
(
"ieee1394"
);
hpsb_packet_cache
=
kmem_cache_create
(
"hpsb_packet"
,
sizeof
(
struct
hpsb_packet
),
0
,
0
,
NULL
,
NULL
);
bus_register
(
&
ieee1394_bus_type
);
init_hpsb_highlevel
();
init_csr
();
if
(
!
disable_nodemgr
)
init_ieee1394_nodemgr
();
else
...
...
@@ -1198,6 +1201,9 @@ static void __exit ieee1394_cleanup(void)
cleanup_ieee1394_nodemgr
();
cleanup_csr
();
bus_unregister
(
&
ieee1394_bus_type
);
kmem_cache_destroy
(
hpsb_packet_cache
);
unregister_chrdev
(
IEEE1394_MAJOR
,
"ieee1394"
);
...
...
@@ -1213,16 +1219,15 @@ module_exit(ieee1394_cleanup);
EXPORT_SYMBOL
(
hpsb_alloc_host
);
EXPORT_SYMBOL
(
hpsb_add_host
);
EXPORT_SYMBOL
(
hpsb_remove_host
);
EXPORT_SYMBOL
(
hpsb_ref_host
);
EXPORT_SYMBOL
(
hpsb_unref_host
);
/** ieee1394_core.c **/
EXPORT_SYMBOL
(
hpsb_speedto_str
);
EXPORT_SYMBOL
(
hpsb_set_packet_complete_task
);
EXPORT_SYMBOL
(
alloc_hpsb
_packet
);
EXPORT_SYMBOL
(
free_hpsb
_packet
);
EXPORT_SYMBOL
(
hpsb_alloc
_packet
);
EXPORT_SYMBOL
(
hpsb_free
_packet
);
EXPORT_SYMBOL
(
hpsb_send_phy_config
);
EXPORT_SYMBOL
(
hpsb_send_packet
);
EXPORT_SYMBOL
(
hpsb_send_packet_and_wait
);
EXPORT_SYMBOL
(
hpsb_reset_bus
);
EXPORT_SYMBOL
(
hpsb_bus_reset
);
EXPORT_SYMBOL
(
hpsb_selfid_received
);
...
...
@@ -1282,6 +1287,7 @@ EXPORT_SYMBOL(hpsb_node_lock);
EXPORT_SYMBOL
(
hpsb_register_protocol
);
EXPORT_SYMBOL
(
hpsb_unregister_protocol
);
EXPORT_SYMBOL
(
ieee1394_bus_type
);
EXPORT_SYMBOL
(
nodemgr_for_each_host
);
/** csr.c **/
EXPORT_SYMBOL
(
hpsb_update_config_rom
);
...
...
drivers/ieee1394/ieee1394_core.h
View file @
d911e854
...
...
@@ -40,11 +40,6 @@ struct hpsb_packet {
unsigned
expect_response
:
1
;
unsigned
no_waiter
:
1
;
/* Data big endianness flag - may vary from request to request. The
* header is always in machine byte order.
* Not really used currently. */
unsigned
data_be
:
1
;
/* Speed to transmit with: 0 = 100Mbps, 1 = 200Mbps, 2 = 400Mbps */
unsigned
speed_code
:
2
;
...
...
@@ -64,9 +59,6 @@ struct hpsb_packet {
struct
hpsb_host
*
host
;
unsigned
int
generation
;
/* Very core internal, don't care. */
struct
semaphore
state_change
;
/* Function (and possible data to pass to it) to call when this
* packet is completed. */
void
(
*
complete_routine
)(
void
*
);
...
...
@@ -90,8 +82,8 @@ static inline struct hpsb_packet *driver_packet(struct list_head *l)
void
abort_timedouts
(
unsigned
long
__opaque
);
void
abort_requests
(
struct
hpsb_host
*
host
);
struct
hpsb_packet
*
alloc_hpsb
_packet
(
size_t
data_size
);
void
free_hpsb
_packet
(
struct
hpsb_packet
*
packet
);
struct
hpsb_packet
*
hpsb_alloc
_packet
(
size_t
data_size
);
void
hpsb_free
_packet
(
struct
hpsb_packet
*
packet
);
/*
...
...
@@ -108,15 +100,23 @@ static inline unsigned int get_hpsb_generation(struct hpsb_host *host)
}
/*
* Send a PHY configuration packet.
* Send a PHY configuration packet, return 0 on success, negative
* errno on failure.
*/
int
hpsb_send_phy_config
(
struct
hpsb_host
*
host
,
int
rootid
,
int
gapcnt
);
/*
* Queue packet for transmitting, return 0 for failure.
* Queue packet for transmitting, return 0 on success, negative errno
* on failure.
*/
int
hpsb_send_packet
(
struct
hpsb_packet
*
packet
);
/*
* Queue packet for transmitting, and block until the transaction
* completes. Return 0 on success, negative errno on failure.
*/
int
hpsb_send_packet_and_wait
(
struct
hpsb_packet
*
packet
);
/* Initiate bus reset on the given host. Returns 1 if bus reset already in
* progress, 0 otherwise. */
int
hpsb_reset_bus
(
struct
hpsb_host
*
host
,
int
type
);
...
...
drivers/ieee1394/ieee1394_transactions.c
View file @
d911e854
...
...
@@ -263,7 +263,7 @@ struct hpsb_packet *hpsb_make_readpacket(struct hpsb_host *host, nodeid_t node,
if
(
length
==
0
)
return
NULL
;
packet
=
alloc_hpsb
_packet
(
length
+
(
length
%
4
?
4
-
(
length
%
4
)
:
0
));
packet
=
hpsb_alloc
_packet
(
length
+
(
length
%
4
?
4
-
(
length
%
4
)
:
0
));
if
(
!
packet
)
return
NULL
;
...
...
@@ -271,7 +271,7 @@ struct hpsb_packet *hpsb_make_readpacket(struct hpsb_host *host, nodeid_t node,
packet
->
node_id
=
node
;
if
(
hpsb_get_tlabel
(
packet
))
{
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
NULL
;
}
...
...
@@ -291,7 +291,7 @@ struct hpsb_packet *hpsb_make_writepacket (struct hpsb_host *host, nodeid_t node
if
(
length
==
0
)
return
NULL
;
packet
=
alloc_hpsb
_packet
(
length
+
(
length
%
4
?
4
-
(
length
%
4
)
:
0
));
packet
=
hpsb_alloc
_packet
(
length
+
(
length
%
4
?
4
-
(
length
%
4
)
:
0
));
if
(
!
packet
)
return
NULL
;
...
...
@@ -302,7 +302,7 @@ struct hpsb_packet *hpsb_make_writepacket (struct hpsb_host *host, nodeid_t node
packet
->
node_id
=
node
;
if
(
hpsb_get_tlabel
(
packet
))
{
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
NULL
;
}
...
...
@@ -325,7 +325,7 @@ struct hpsb_packet *hpsb_make_streampacket(struct hpsb_host *host, u8 *buffer, i
if
(
length
==
0
)
return
NULL
;
packet
=
alloc_hpsb
_packet
(
length
+
(
length
%
4
?
4
-
(
length
%
4
)
:
0
));
packet
=
hpsb_alloc
_packet
(
length
+
(
length
%
4
?
4
-
(
length
%
4
)
:
0
));
if
(
!
packet
)
return
NULL
;
...
...
@@ -335,7 +335,7 @@ struct hpsb_packet *hpsb_make_streampacket(struct hpsb_host *host, u8 *buffer, i
packet
->
host
=
host
;
if
(
hpsb_get_tlabel
(
packet
))
{
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
NULL
;
}
...
...
@@ -353,13 +353,13 @@ struct hpsb_packet *hpsb_make_lockpacket(struct hpsb_host *host, nodeid_t node,
struct
hpsb_packet
*
p
;
u32
length
;
p
=
alloc_hpsb
_packet
(
8
);
p
=
hpsb_alloc
_packet
(
8
);
if
(
!
p
)
return
NULL
;
p
->
host
=
host
;
p
->
node_id
=
node
;
if
(
hpsb_get_tlabel
(
p
))
{
free_hpsb
_packet
(
p
);
hpsb_free
_packet
(
p
);
return
NULL
;
}
...
...
@@ -390,13 +390,13 @@ struct hpsb_packet *hpsb_make_lock64packet(struct hpsb_host *host, nodeid_t node
struct
hpsb_packet
*
p
;
u32
length
;
p
=
alloc_hpsb
_packet
(
16
);
p
=
hpsb_alloc
_packet
(
16
);
if
(
!
p
)
return
NULL
;
p
->
host
=
host
;
p
->
node_id
=
node
;
if
(
hpsb_get_tlabel
(
p
))
{
free_hpsb
_packet
(
p
);
hpsb_free
_packet
(
p
);
return
NULL
;
}
...
...
@@ -429,7 +429,7 @@ struct hpsb_packet *hpsb_make_phypacket(struct hpsb_host *host,
{
struct
hpsb_packet
*
p
;
p
=
alloc_hpsb
_packet
(
0
);
p
=
hpsb_alloc
_packet
(
0
);
if
(
!
p
)
return
NULL
;
p
->
host
=
host
;
...
...
@@ -444,7 +444,7 @@ struct hpsb_packet *hpsb_make_isopacket(struct hpsb_host *host,
{
struct
hpsb_packet
*
p
;
p
=
alloc_hpsb
_packet
(
length
);
p
=
hpsb_alloc
_packet
(
length
);
if
(
!
p
)
return
NULL
;
p
->
host
=
host
;
...
...
@@ -478,13 +478,10 @@ int hpsb_read(struct hpsb_host *host, nodeid_t node, unsigned int generation,
}
packet
->
generation
=
generation
;
if
(
!
hpsb_send_packet
(
packet
))
{
retval
=
-
EINVAL
;
retval
=
hpsb_send_packet_and_wait
(
packet
);
if
(
retval
<
0
)
goto
hpsb_read_fail
;
}
down
(
&
packet
->
state_change
);
down
(
&
packet
->
state_change
);
retval
=
hpsb_packet_success
(
packet
);
if
(
retval
==
0
)
{
...
...
@@ -497,7 +494,7 @@ int hpsb_read(struct hpsb_host *host, nodeid_t node, unsigned int generation,
hpsb_read_fail:
hpsb_free_tlabel
(
packet
);
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
retval
;
}
...
...
@@ -520,18 +517,15 @@ int hpsb_write(struct hpsb_host *host, nodeid_t node, unsigned int generation,
return
-
ENOMEM
;
packet
->
generation
=
generation
;
if
(
!
hpsb_send_packet
(
packet
))
{
retval
=
-
EINVAL
;
retval
=
hpsb_send_packet_and_wait
(
packet
);
if
(
retval
<
0
)
goto
hpsb_write_fail
;
}
down
(
&
packet
->
state_change
);
down
(
&
packet
->
state_change
);
retval
=
hpsb_packet_success
(
packet
);
hpsb_write_fail:
hpsb_free_tlabel
(
packet
);
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
retval
;
}
...
...
@@ -550,12 +544,10 @@ int hpsb_lock(struct hpsb_host *host, nodeid_t node, unsigned int generation,
return
-
ENOMEM
;
packet
->
generation
=
generation
;
if
(
!
hpsb_send_packet
(
packet
))
{
retval
=
-
EINVAL
;
retval
=
hpsb_send_packet_and_wait
(
packet
);
if
(
retval
<
0
)
goto
hpsb_lock_fail
;
}
down
(
&
packet
->
state_change
);
down
(
&
packet
->
state_change
);
retval
=
hpsb_packet_success
(
packet
);
if
(
retval
==
0
)
{
...
...
@@ -564,7 +556,7 @@ int hpsb_lock(struct hpsb_host *host, nodeid_t node, unsigned int generation,
hpsb_lock_fail:
hpsb_free_tlabel
(
packet
);
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
retval
;
}
...
...
@@ -582,12 +574,10 @@ int hpsb_lock64(struct hpsb_host *host, nodeid_t node, unsigned int generation,
return
-
ENOMEM
;
packet
->
generation
=
generation
;
if
(
!
hpsb_send_packet
(
packet
))
{
retval
=
-
EINVAL
;
retval
=
hpsb_send_packet_and_wait
(
packet
);
if
(
retval
<
0
)
goto
hpsb_lock64_fail
;
}
down
(
&
packet
->
state_change
);
down
(
&
packet
->
state_change
);
retval
=
hpsb_packet_success
(
packet
);
if
(
retval
==
0
)
...
...
@@ -595,7 +585,7 @@ int hpsb_lock64(struct hpsb_host *host, nodeid_t node, unsigned int generation,
hpsb_lock64_fail:
hpsb_free_tlabel
(
packet
);
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
return
retval
;
}
...
...
@@ -626,10 +616,9 @@ int hpsb_send_gasp(struct hpsb_host *host, int channel, unsigned int generation,
packet
->
no_waiter
=
1
;
if
(
!
hpsb_send_packet
(
packet
))
{
free_hpsb_packet
(
packet
);
retval
=
-
EINVAL
;
}
retval
=
hpsb_send_packet
(
packet
);
if
(
retval
<
0
)
hpsb_free_packet
(
packet
);
return
retval
;
}
drivers/ieee1394/iso.c
View file @
d911e854
...
...
@@ -40,6 +40,7 @@ static struct hpsb_iso* hpsb_iso_common_init(struct hpsb_host *host, enum hpsb_i
unsigned
int
data_buf_size
,
unsigned
int
buf_packets
,
int
channel
,
int
dma_mode
,
int
irq_interval
,
void
(
*
callback
)(
struct
hpsb_iso
*
))
{
...
...
@@ -58,8 +59,13 @@ static struct hpsb_iso* hpsb_iso_common_init(struct hpsb_host *host, enum hpsb_i
if
(
buf_packets
<
2
)
buf_packets
=
2
;
if
(
irq_interval
<
1
||
irq_interval
>
buf_packets
/
2
)
irq_interval
=
buf_packets
/
2
;
if
((
dma_mode
<
HPSB_ISO_DMA_DEFAULT
)
||
(
dma_mode
>
HPSB_ISO_DMA_PACKET_PER_BUFFER
))
dma_mode
=
HPSB_ISO_DMA_DEFAULT
;
if
(
irq_interval
==
0
)
/* really interrupt for each packet*/
irq_interval
=
1
;
else
if
((
irq_interval
<
0
)
||
(
irq_interval
>
buf_packets
/
4
))
irq_interval
=
buf_packets
/
4
;
if
(
channel
<
-
1
||
channel
>=
64
)
return
NULL
;
...
...
@@ -83,6 +89,7 @@ static struct hpsb_iso* hpsb_iso_common_init(struct hpsb_host *host, enum hpsb_i
init_waitqueue_head
(
&
iso
->
waitq
);
iso
->
channel
=
channel
;
iso
->
irq_interval
=
irq_interval
;
iso
->
dma_mode
=
dma_mode
;
dma_region_init
(
&
iso
->
data_buf
);
iso
->
buf_size
=
round_up_to_page
(
data_buf_size
);
iso
->
buf_packets
=
buf_packets
;
...
...
@@ -136,7 +143,7 @@ struct hpsb_iso* hpsb_iso_xmit_init(struct hpsb_host *host,
{
struct
hpsb_iso
*
iso
=
hpsb_iso_common_init
(
host
,
HPSB_ISO_XMIT
,
data_buf_size
,
buf_packets
,
channel
,
irq_interval
,
callback
);
channel
,
HPSB_ISO_DMA_DEFAULT
,
irq_interval
,
callback
);
if
(
!
iso
)
return
NULL
;
...
...
@@ -158,12 +165,13 @@ struct hpsb_iso* hpsb_iso_recv_init(struct hpsb_host *host,
unsigned
int
data_buf_size
,
unsigned
int
buf_packets
,
int
channel
,
int
dma_mode
,
int
irq_interval
,
void
(
*
callback
)(
struct
hpsb_iso
*
))
{
struct
hpsb_iso
*
iso
=
hpsb_iso_common_init
(
host
,
HPSB_ISO_RECV
,
data_buf_size
,
buf_packets
,
channel
,
irq_interval
,
callback
);
channel
,
dma_mode
,
irq_interval
,
callback
);
if
(
!
iso
)
return
NULL
;
...
...
drivers/ieee1394/iso.h
View file @
d911e854
...
...
@@ -51,6 +51,14 @@ struct hpsb_iso_packet_info {
enum
hpsb_iso_type
{
HPSB_ISO_RECV
=
0
,
HPSB_ISO_XMIT
=
1
};
/* The mode of the dma when receiving iso data. Must be supported by chip */
enum
raw1394_iso_dma_recv_mode
{
HPSB_ISO_DMA_DEFAULT
=
-
1
,
HPSB_ISO_DMA_OLD_ABI
=
0
,
HPSB_ISO_DMA_BUFFERFILL
=
1
,
HPSB_ISO_DMA_PACKET_PER_BUFFER
=
2
};
struct
hpsb_iso
{
enum
hpsb_iso_type
type
;
...
...
@@ -68,6 +76,8 @@ struct hpsb_iso {
int
speed
;
/* IEEE1394_SPEED_100, 200, or 400 */
int
channel
;
/* -1 if multichannel */
int
dma_mode
;
/* dma receive mode */
/* greatest # of packets between interrupts - controls
the maximum latency of the buffer */
...
...
@@ -139,6 +149,7 @@ struct hpsb_iso* hpsb_iso_recv_init(struct hpsb_host *host,
unsigned
int
data_buf_size
,
unsigned
int
buf_packets
,
int
channel
,
int
dma_mode
,
int
irq_interval
,
void
(
*
callback
)(
struct
hpsb_iso
*
));
...
...
drivers/ieee1394/nodemgr.c
View file @
d911e854
...
...
@@ -516,8 +516,7 @@ static struct device nodemgr_dev_template_ne = {
.
driver_data
=
&
nodemgr_driverdata_ne
,
};
static
struct
device
nodemgr_dev_template_host
=
{
struct
device
nodemgr_dev_template_host
=
{
.
bus
=
&
ieee1394_bus_type
,
.
release
=
nodemgr_release_host
,
.
driver
=
&
nodemgr_driver_host
,
...
...
@@ -1255,36 +1254,6 @@ void hpsb_unregister_protocol(struct hpsb_protocol_driver *driver)
}
/* Searches the list of ud's that match a ne as the parent. If the ud has
* a driver associated with it, we call that driver's update function
* with the ud as the argument. */
static
int
nodemgr_driver_search_cb
(
struct
device
*
dev
,
void
*
__data
)
{
struct
node_entry
*
ne
=
__data
;
struct
unit_directory
*
ud
;
if
(
dev
->
driver_data
==
&
nodemgr_driverdata_ne
||
dev
->
driver_data
==
&
nodemgr_driverdata_host
)
return
0
;
ud
=
container_of
(
dev
,
struct
unit_directory
,
device
);
if
(
&
ne
->
device
!=
ud
->
device
.
parent
)
return
0
;
if
(
ud
->
device
.
driver
)
{
struct
hpsb_protocol_driver
*
pdrv
;
pdrv
=
container_of
(
ud
->
device
.
driver
,
struct
hpsb_protocol_driver
,
driver
);
if
(
pdrv
->
update
)
pdrv
->
update
(
ud
);
}
return
0
;
}
/*
* This function updates nodes that were present on the bus before the
* reset and still are after the reset. The nodeid and the config rom
...
...
@@ -1480,9 +1449,24 @@ static int nodemgr_probe_ne_cb(struct device *dev, void *__data)
ne
->
needs_probe
=
0
;
return
1
;
}
else
{
struct
list_head
*
lh
;
/* Update unit_dirs with attached drivers */
bus_for_each_dev
(
&
ieee1394_bus_type
,
NULL
,
ne
,
nodemgr_driver_search_cb
);
list_for_each
(
lh
,
&
dev
->
children
)
{
struct
unit_directory
*
ud
;
ud
=
container_of
(
list_to_dev
(
lh
),
struct
unit_directory
,
device
);
if
(
ud
->
device
.
driver
)
{
struct
hpsb_protocol_driver
*
pdrv
;
pdrv
=
container_of
(
ud
->
device
.
driver
,
struct
hpsb_protocol_driver
,
driver
);
if
(
pdrv
->
update
)
pdrv
->
update
(
ud
);
}
}
}
return
0
;
}
...
...
@@ -1664,7 +1648,6 @@ static int nodemgr_host_thread(void *__hi)
allow_signal
(
SIGTERM
);
/* Setup our device-model entries */
device_register
(
&
host
->
device
);
nodemgr_create_host_dev_files
(
host
);
/* Sit and wait for a signal to probe the nodes on the bus. This
...
...
@@ -1753,6 +1736,34 @@ struct node_entry *hpsb_nodeid_get_entry(struct hpsb_host *host, nodeid_t nodeid
return
ne
;
}
struct
for_each_host_struct
{
int
(
*
cb
)(
struct
hpsb_host
*
,
void
*
);
void
*
data
;
};
static
int
nodemgr_for_each_host_cb
(
struct
device
*
dev
,
void
*
__data
)
{
struct
for_each_host_struct
*
host_data
=
__data
;
struct
hpsb_host
*
host
;
if
(
dev
->
driver_data
!=
&
nodemgr_driverdata_host
)
return
0
;
host
=
container_of
(
dev
,
struct
hpsb_host
,
device
);
return
host_data
->
cb
(
host
,
host_data
->
data
);
}
int
nodemgr_for_each_host
(
void
*
__data
,
int
(
*
cb
)(
struct
hpsb_host
*
,
void
*
))
{
struct
for_each_host_struct
host_data
;
host_data
.
cb
=
cb
;
host_data
.
data
=
__data
;
return
bus_for_each_dev
(
&
ieee1394_bus_type
,
NULL
,
&
host_data
,
nodemgr_for_each_host_cb
);
}
/* The following four convenience functions use a struct node_entry
* for addressing a node on the bus. They are intended for use by any
* process context, not just the nodemgr thread, so we need to be a
...
...
@@ -1821,11 +1832,6 @@ static void nodemgr_add_host(struct hpsb_host *host)
init_completion
(
&
hi
->
exited
);
sema_init
(
&
hi
->
reset_sem
,
0
);
memcpy
(
&
host
->
device
,
&
nodemgr_dev_template_host
,
sizeof
(
host
->
device
));
host
->
device
.
parent
=
&
host
->
pdev
->
dev
;
snprintf
(
host
->
device
.
bus_id
,
BUS_ID_SIZE
,
"fw-host%d"
,
host
->
id
);
sprintf
(
hi
->
daemon_name
,
"knodemgrd_%d"
,
host
->
id
);
hi
->
pid
=
kernel_thread
(
nodemgr_host_thread
,
hi
,
CLONE_KERNEL
);
...
...
@@ -1879,7 +1885,6 @@ static struct hpsb_highlevel nodemgr_highlevel = {
void
init_ieee1394_nodemgr
(
void
)
{
bus_register
(
&
ieee1394_bus_type
);
driver_register
(
&
nodemgr_driver_host
);
driver_register
(
&
nodemgr_driver_ne
);
...
...
@@ -1892,5 +1897,4 @@ void cleanup_ieee1394_nodemgr(void)
driver_unregister
(
&
nodemgr_driver_ne
);
driver_unregister
(
&
nodemgr_driver_host
);
bus_unregister
(
&
ieee1394_bus_type
);
}
drivers/ieee1394/nodemgr.h
View file @
d911e854
...
...
@@ -219,7 +219,16 @@ int hpsb_node_lock(struct node_entry *ne, u64 addr,
int
extcode
,
quadlet_t
*
data
,
quadlet_t
arg
);
/* Iterate the hosts, calling a given function with supplied data for each
* host. */
int
nodemgr_for_each_host
(
void
*
__data
,
int
(
*
cb
)(
struct
hpsb_host
*
,
void
*
));
void
init_ieee1394_nodemgr
(
void
);
void
cleanup_ieee1394_nodemgr
(
void
);
/* The template for a host device */
extern
struct
device
nodemgr_dev_template_host
;
#endif
/* _IEEE1394_NODEMGR_H */
drivers/ieee1394/ohci1394.c
View file @
d911e854
...
...
@@ -134,7 +134,7 @@
#ifdef OHCI1394_DEBUG
#define DBGMSG(card, fmt, args...) \
printk(KERN_INFO "%s
_
%d: " fmt "\n" , OHCI1394_DRIVER_NAME, card , ## args)
printk(KERN_INFO "%s
: fw-host
%d: " fmt "\n" , OHCI1394_DRIVER_NAME, card , ## args)
#else
#define DBGMSG(card, fmt, args...)
#endif
...
...
@@ -158,10 +158,10 @@ printk(level "%s: " fmt "\n" , OHCI1394_DRIVER_NAME , ## args)
/* print card specific information */
#define PRINT(level, card, fmt, args...) \
printk(level "%s
_
%d: " fmt "\n" , OHCI1394_DRIVER_NAME, card , ## args)
printk(level "%s
: fw-host
%d: " fmt "\n" , OHCI1394_DRIVER_NAME, card , ## args)
static
char
version
[]
__devinitdata
=
"$Rev: 10
45
$ Ben Collins <bcollins@debian.org>"
;
"$Rev: 10
87
$ Ben Collins <bcollins@debian.org>"
;
/* Module Parameters */
static
int
phys_dma
=
1
;
...
...
@@ -845,7 +845,7 @@ static int ohci_transmit(struct hpsb_host *host, struct hpsb_packet *packet)
PRINT
(
KERN_ERR
,
ohci
->
id
,
"Transmit packet size %Zd is too big"
,
packet
->
data_size
);
return
0
;
return
-
EOVERFLOW
;
}
/* Decide whether we have an iso, a request, or a response packet */
...
...
@@ -862,7 +862,7 @@ static int ohci_transmit(struct hpsb_host *host, struct hpsb_packet *packet)
if
(
in_interrupt
())
{
PRINT
(
KERN_ERR
,
ohci
->
id
,
"legacy IT context cannot be initialized during interrupt"
);
return
0
;
return
-
EINVAL
;
}
if
(
alloc_dma_trm_ctx
(
ohci
,
&
ohci
->
it_legacy_context
,
...
...
@@ -870,7 +870,7 @@ static int ohci_transmit(struct hpsb_host *host, struct hpsb_packet *packet)
OHCI1394_IsoXmitContextBase
)
<
0
)
{
PRINT
(
KERN_ERR
,
ohci
->
id
,
"error initializing legacy IT context"
);
return
0
;
return
-
ENOMEM
;
}
initialize_dma_trm_ctx
(
&
ohci
->
it_legacy_context
);
...
...
@@ -890,7 +890,7 @@ static int ohci_transmit(struct hpsb_host *host, struct hpsb_packet *packet)
spin_unlock_irqrestore
(
&
d
->
lock
,
flags
);
return
1
;
return
0
;
}
static
int
ohci_devctl
(
struct
hpsb_host
*
host
,
enum
devctl_cmd
cmd
,
int
arg
)
...
...
@@ -1113,8 +1113,8 @@ struct ohci_iso_recv {
struct
ohci1394_iso_tasklet
task
;
int
task_active
;
enum
{
BUFFER_FILL_MODE
,
PACKET_PER_BUFFER_MODE
}
dma_mode
;
enum
{
BUFFER_FILL_MODE
=
0
,
PACKET_PER_BUFFER_MODE
=
1
}
dma_mode
;
/* memory and PCI mapping for the DMA descriptors */
struct
dma_prog_region
prog
;
...
...
@@ -1175,7 +1175,8 @@ static int ohci_iso_recv_init(struct hpsb_iso *iso)
/* use buffer-fill mode, unless irq_interval is 1
(note: multichannel requires buffer-fill) */
if
(
iso
->
irq_interval
==
1
&&
iso
->
channel
!=
-
1
)
{
if
(((
iso
->
irq_interval
==
1
&&
iso
->
dma_mode
==
HPSB_ISO_DMA_OLD_ABI
)
||
iso
->
dma_mode
==
HPSB_ISO_DMA_PACKET_PER_BUFFER
)
&&
iso
->
channel
!=
-
1
)
{
recv
->
dma_mode
=
PACKET_PER_BUFFER_MODE
;
}
else
{
recv
->
dma_mode
=
BUFFER_FILL_MODE
;
...
...
@@ -1194,8 +1195,11 @@ static int ohci_iso_recv_init(struct hpsb_iso *iso)
}
/* iso->irq_interval is in packets - translate that to blocks */
/* (err, sort of... 1 is always the safest value) */
recv
->
block_irq_interval
=
iso
->
irq_interval
/
recv
->
nblocks
;
if
(
iso
->
irq_interval
==
1
)
recv
->
block_irq_interval
=
1
;
else
recv
->
block_irq_interval
=
iso
->
irq_interval
*
((
recv
->
nblocks
+
1
)
/
iso
->
buf_packets
);
if
(
recv
->
block_irq_interval
*
4
>
recv
->
nblocks
)
recv
->
block_irq_interval
=
recv
->
nblocks
/
4
;
if
(
recv
->
block_irq_interval
<
1
)
...
...
@@ -1205,6 +1209,10 @@ static int ohci_iso_recv_init(struct hpsb_iso *iso)
int
max_packet_size
;
recv
->
nblocks
=
iso
->
buf_packets
;
recv
->
block_irq_interval
=
iso
->
irq_interval
;
if
(
recv
->
block_irq_interval
*
4
>
iso
->
buf_packets
)
recv
->
block_irq_interval
=
iso
->
buf_packets
/
4
;
if
(
recv
->
block_irq_interval
<
1
)
recv
->
block_irq_interval
=
1
;
/* choose a buffer stride */
...
...
@@ -3271,7 +3279,6 @@ do { \
static
int
__devinit
ohci1394_pci_probe
(
struct
pci_dev
*
dev
,
const
struct
pci_device_id
*
ent
)
{
static
unsigned
int
card_id_counter
=
0
;
static
int
version_printed
=
0
;
struct
hpsb_host
*
host
;
...
...
@@ -3282,15 +3289,14 @@ static int __devinit ohci1394_pci_probe(struct pci_dev *dev,
PRINT_G
(
KERN_INFO
,
"%s"
,
version
);
if
(
pci_enable_device
(
dev
))
FAIL
(
-
ENXIO
,
"Failed to enable OHCI hardware %d"
,
card_id_counter
++
);
FAIL
(
-
ENXIO
,
"Failed to enable OHCI hardware"
);
pci_set_master
(
dev
);
host
=
hpsb_alloc_host
(
&
ohci1394_driver
,
sizeof
(
struct
ti_ohci
));
host
=
hpsb_alloc_host
(
&
ohci1394_driver
,
sizeof
(
struct
ti_ohci
)
,
&
dev
->
dev
);
if
(
!
host
)
FAIL
(
-
ENOMEM
,
"Failed to allocate host structure"
);
ohci
=
host
->
hostdata
;
ohci
->
id
=
card_id_counter
++
;
ohci
->
id
=
host
->
id
;
ohci
->
dev
=
dev
;
ohci
->
host
=
host
;
ohci
->
init_state
=
OHCI_INIT_ALLOC_HOST
;
...
...
@@ -3462,11 +3468,14 @@ static int __devinit ohci1394_pci_probe(struct pci_dev *dev,
static
void
ohci1394_pci_remove
(
struct
pci_dev
*
pdev
)
{
struct
ti_ohci
*
ohci
;
struct
device
*
dev
;
ohci
=
pci_get_drvdata
(
pdev
);
if
(
!
ohci
)
return
;
dev
=
get_device
(
&
ohci
->
host
->
device
);
switch
(
ohci
->
init_state
)
{
case
OHCI_INIT_DONE
:
hpsb_remove_host
(
ohci
->
host
);
...
...
@@ -3489,7 +3498,7 @@ static void ohci1394_pci_remove(struct pci_dev *pdev)
/* Disable IRM Contender */
set_phy_reg
(
ohci
,
4
,
~
0xc0
&
get_phy_reg
(
ohci
,
4
));
/* Clear link control register */
reg_write
(
ohci
,
OHCI1394_LinkControlClear
,
0xffffffff
);
...
...
@@ -3521,7 +3530,7 @@ static void ohci1394_pci_remove(struct pci_dev *pdev)
ohci
->
selfid_buf_cpu
,
ohci
->
selfid_buf_bus
);
OHCI_DMA_FREE
(
"consistent selfid_buf"
);
case
OHCI_INIT_HAVE_CONFIG_ROM_BUFFER
:
pci_free_consistent
(
ohci
->
dev
,
OHCI_CONFIG_ROM_LEN
,
ohci
->
csr_config_rom_cpu
,
...
...
@@ -3555,8 +3564,10 @@ static void ohci1394_pci_remove(struct pci_dev *pdev)
case
OHCI_INIT_ALLOC_HOST
:
pci_set_drvdata
(
ohci
->
dev
,
NULL
);
hpsb_unref_host
(
ohci
->
host
);
}
if
(
dev
)
put_device
(
dev
);
}
...
...
drivers/ieee1394/oui.db
View file @
d911e854
...
...
@@ -1612,7 +1612,7 @@
000658 Helmut Fischer GmbH & Co. KG
000659 EAL (Apeldoorn) B.V.
00065A Strix Systems
00065B Dell
Inc
.
00065B Dell
Computer Corp
.
00065C Malachite Technologies, Inc.
00065D Heidelberg Web Systems
00065E Photuris, Inc.
...
...
@@ -3965,7 +3965,7 @@
00B0C2 Cisco Systems, Inc.
00B0C7 Tellabs Operations, Inc.
00B0CE TECHNOLOGY RESCUE
00B0D0 Dell
Inc
.
00B0D0 Dell
Computer Corp
.
00B0DB Nextcell, Inc.
00B0DF Reliable Data Technology, Inc.
00B0E7 British Federal Ltd.
...
...
@@ -4054,7 +4054,7 @@
00C04C DEPARTMENT OF FOREIGN AFFAIRS
00C04D MITEC, INC.
00C04E COMTROL CORPORATION
00C04F D
ell Inc.
00C04F D
ELL COMPUTER CORPORATION
00C050 TOYO DENKI SEIZO K.K.
00C051 ADVANCED INTEGRATION RESEARCH
00C052 BURR-BROWN
...
...
drivers/ieee1394/pcilynx.c
View file @
d911e854
...
...
@@ -521,10 +521,6 @@ static void send_next(struct ti_lynx *lynx, int what)
break
;
}
if
(
!
packet
->
data_be
)
{
pcl
.
buffer
[
1
].
control
|=
PCL_BIGENDIAN
;
}
put_pcl
(
lynx
,
d
->
pcl
,
&
pcl
);
run_pcl
(
lynx
,
d
->
pcl_start
,
d
->
channel
);
}
...
...
@@ -540,7 +536,7 @@ static int lynx_transmit(struct hpsb_host *host, struct hpsb_packet *packet)
if
(
packet
->
data_size
>=
4096
)
{
PRINT
(
KERN_ERR
,
lynx
->
id
,
"transmit packet data too big (%Zd)"
,
packet
->
data_size
);
return
0
;
return
-
EOVERFLOW
;
}
switch
(
packet
->
type
)
{
...
...
@@ -554,7 +550,7 @@ static int lynx_transmit(struct hpsb_host *host, struct hpsb_packet *packet)
default:
PRINT
(
KERN_ERR
,
lynx
->
id
,
"invalid packet type %d"
,
packet
->
type
);
return
0
;
return
-
EINVAL
;
}
if
(
packet
->
tcode
==
TCODE_WRITEQ
...
...
@@ -570,7 +566,7 @@ static int lynx_transmit(struct hpsb_host *host, struct hpsb_packet *packet)
spin_unlock_irqrestore
(
&
d
->
queue_lock
,
flags
);
return
1
;
return
0
;
}
...
...
@@ -1120,7 +1116,7 @@ static ssize_t mem_read(struct file *file, char *buffer, size_t count,
retval
=
copy_to_user
(
buffer
,
md
->
lynx
->
mem_dma_buffer
,
count
);
up
(
&
md
->
lynx
->
mem_dma_mutex
);
if
(
retval
<
0
)
return
retval
;
if
(
retval
)
return
-
EFAULT
;
*
offset
+=
count
;
return
count
;
}
...
...
@@ -1141,14 +1137,17 @@ static ssize_t mem_write(struct file *file, const char *buffer, size_t count,
/* FIXME: dereferencing pointers to PCI mem doesn't work everywhere */
switch
(
md
->
type
)
{
case
aux
:
copy_from_user
(
md
->
lynx
->
aux_port
+
(
*
offset
),
buffer
,
count
);
if
(
copy_from_user
(
md
->
lynx
->
aux_port
+
(
*
offset
),
buffer
,
count
))
return
-
EFAULT
;
break
;
case
ram
:
copy_from_user
(
md
->
lynx
->
local_ram
+
(
*
offset
),
buffer
,
count
);
if
(
copy_from_user
(
md
->
lynx
->
local_ram
+
(
*
offset
),
buffer
,
count
))
return
-
EFAULT
;
break
;
case
rom
:
/* the ROM may be writeable */
copy_from_user
(
md
->
lynx
->
local_rom
+
(
*
offset
),
buffer
,
count
);
if
(
copy_from_user
(
md
->
lynx
->
local_rom
+
(
*
offset
),
buffer
,
count
))
return
-
EFAULT
;
break
;
}
...
...
@@ -1443,12 +1442,15 @@ static void iso_rcv_bh(struct ti_lynx *lynx)
static
void
remove_card
(
struct
pci_dev
*
dev
)
{
struct
ti_lynx
*
lynx
;
struct
device
*
lynx_dev
;
int
i
;
lynx
=
pci_get_drvdata
(
dev
);
if
(
!
lynx
)
return
;
pci_set_drvdata
(
dev
,
NULL
);
lynx_dev
=
get_device
(
&
lynx
->
host
->
device
);
switch
(
lynx
->
state
)
{
case
is_host
:
reg_write
(
lynx
,
PCI_INT_ENABLE
,
0
);
...
...
@@ -1498,7 +1500,9 @@ static void remove_card(struct pci_dev *dev)
}
tasklet_kill
(
&
lynx
->
iso_rcv
.
tq
);
hpsb_unref_host
(
lynx
->
host
);
if
(
lynx_dev
)
put_device
(
lynx_dev
);
}
...
...
@@ -1535,7 +1539,7 @@ static int __devinit add_card(struct pci_dev *dev,
error
=
-
ENOMEM
;
host
=
hpsb_alloc_host
(
&
lynx_driver
,
sizeof
(
struct
ti_lynx
));
host
=
hpsb_alloc_host
(
&
lynx_driver
,
sizeof
(
struct
ti_lynx
)
,
&
dev
->
dev
);
if
(
!
host
)
FAIL
(
"failed to allocate control structure memory"
);
lynx
=
host
->
hostdata
;
...
...
drivers/ieee1394/raw1394.c
View file @
d911e854
...
...
@@ -128,7 +128,7 @@ static void free_pending_request(struct pending_request *req)
}
else
if
(
req
->
free_data
)
{
kfree
(
req
->
data
);
}
free_hpsb
_packet
(
req
->
packet
);
hpsb_free
_packet
(
req
->
packet
);
kfree
(
req
);
}
...
...
@@ -558,7 +558,7 @@ static int state_initialized(struct file_info *fi, struct pending_request *req)
lh
=
lh
->
next
;
}
hi
=
list_entry
(
lh
,
struct
host_info
,
list
);
hpsb_ref_host
(
hi
->
host
);
// XXX Need to handle failure case
get_device
(
&
hi
->
host
->
device
);
// XXX Need to handle failure case
list_add_tail
(
&
fi
->
list
,
&
hi
->
file_info_list
);
fi
->
host
=
hi
->
host
;
fi
->
state
=
connected
;
...
...
@@ -782,7 +782,7 @@ static int handle_async_request(struct file_info *fi,
packet
->
generation
=
req
->
req
.
generation
;
if
(
!
hpsb_send_packet
(
packet
)
)
{
if
(
hpsb_send_packet
(
packet
)
<
0
)
{
req
->
req
.
error
=
RAW1394_ERROR_SEND_ERROR
;
req
->
req
.
length
=
0
;
hpsb_free_tlabel
(
packet
);
...
...
@@ -823,7 +823,7 @@ static int handle_iso_send(struct file_info *fi, struct pending_request *req,
/* Update the generation of the packet just before sending. */
packet
->
generation
=
req
->
req
.
generation
;
if
(
!
hpsb_send_packet
(
packet
)
)
{
if
(
hpsb_send_packet
(
packet
)
<
0
)
{
req
->
req
.
error
=
RAW1394_ERROR_SEND_ERROR
;
queue_complete_req
(
req
);
}
...
...
@@ -846,7 +846,7 @@ static int handle_async_send(struct file_info *fi, struct pending_request *req)
return
sizeof
(
struct
raw1394_request
);
}
packet
=
alloc_hpsb
_packet
(
req
->
req
.
length
-
header_length
);
packet
=
hpsb_alloc
_packet
(
req
->
req
.
length
-
header_length
);
req
->
packet
=
packet
;
if
(
!
packet
)
return
-
ENOMEM
;
...
...
@@ -885,7 +885,7 @@ static int handle_async_send(struct file_info *fi, struct pending_request *req)
/* Update the generation of the packet just before sending. */
packet
->
generation
=
req
->
req
.
generation
;
if
(
!
hpsb_send_packet
(
packet
)
)
{
if
(
hpsb_send_packet
(
packet
)
<
0
)
{
req
->
req
.
error
=
RAW1394_ERROR_SEND_ERROR
;
queue_complete_req
(
req
);
}
...
...
@@ -1797,6 +1797,106 @@ static int arm_unregister(struct file_info *fi, struct pending_request *req)
return
sizeof
(
struct
raw1394_request
);
}
/* Copy data from ARM buffer(s) to user buffer. */
static
int
arm_get_buf
(
struct
file_info
*
fi
,
struct
pending_request
*
req
)
{
struct
arm_addr
*
arm_addr
=
NULL
;
unsigned
long
flags
;
unsigned
long
offset
;
struct
list_head
*
entry
;
DBGMSG
(
"arm_get_buf "
"addr(Offset): %04X %08X length: %u"
,
(
u32
)
((
req
->
req
.
address
>>
32
)
&
0xFFFF
),
(
u32
)
(
req
->
req
.
address
&
0xFFFFFFFF
),
(
u32
)
req
->
req
.
length
);
spin_lock_irqsave
(
&
host_info_lock
,
flags
);
entry
=
fi
->
addr_list
.
next
;
while
(
entry
!=
&
(
fi
->
addr_list
))
{
arm_addr
=
list_entry
(
entry
,
struct
arm_addr
,
addr_list
);
if
((
arm_addr
->
start
<=
req
->
req
.
address
)
&&
(
arm_addr
->
end
>
req
->
req
.
address
))
{
if
(
req
->
req
.
address
+
req
->
req
.
length
<=
arm_addr
->
end
)
{
offset
=
req
->
req
.
address
-
arm_addr
->
start
;
DBGMSG
(
"arm_get_buf copy_to_user( %08X, %08X, %u )"
,
(
u32
)
req
->
req
.
recvb
,
(
u32
)
(
arm_addr
->
addr_space_buffer
+
offset
),
(
u32
)
req
->
req
.
length
);
if
(
copy_to_user
(
int2ptr
(
req
->
req
.
recvb
),
arm_addr
->
addr_space_buffer
+
offset
,
req
->
req
.
length
))
{
spin_unlock_irqrestore
(
&
host_info_lock
,
flags
);
return
(
-
EFAULT
);
}
spin_unlock_irqrestore
(
&
host_info_lock
,
flags
);
free_pending_request
(
req
);
/* we have to free the request, because we queue no response, and therefore nobody will free it */
return
sizeof
(
struct
raw1394_request
);
}
else
{
DBGMSG
(
"arm_get_buf request exceeded mapping"
);
spin_unlock_irqrestore
(
&
host_info_lock
,
flags
);
return
(
-
EINVAL
);
}
}
entry
=
entry
->
next
;
}
spin_unlock_irqrestore
(
&
host_info_lock
,
flags
);
return
(
-
EINVAL
);
}
/* Copy data from user buffer to ARM buffer(s). */
static
int
arm_set_buf
(
struct
file_info
*
fi
,
struct
pending_request
*
req
)
{
struct
arm_addr
*
arm_addr
=
NULL
;
unsigned
long
flags
;
unsigned
long
offset
;
struct
list_head
*
entry
;
DBGMSG
(
"arm_set_buf "
"addr(Offset): %04X %08X length: %u"
,
(
u32
)
((
req
->
req
.
address
>>
32
)
&
0xFFFF
),
(
u32
)
(
req
->
req
.
address
&
0xFFFFFFFF
),
(
u32
)
req
->
req
.
length
);
spin_lock_irqsave
(
&
host_info_lock
,
flags
);
entry
=
fi
->
addr_list
.
next
;
while
(
entry
!=
&
(
fi
->
addr_list
))
{
arm_addr
=
list_entry
(
entry
,
struct
arm_addr
,
addr_list
);
if
((
arm_addr
->
start
<=
req
->
req
.
address
)
&&
(
arm_addr
->
end
>
req
->
req
.
address
))
{
if
(
req
->
req
.
address
+
req
->
req
.
length
<=
arm_addr
->
end
)
{
offset
=
req
->
req
.
address
-
arm_addr
->
start
;
DBGMSG
(
"arm_set_buf copy_from_user( %08X, %08X, %u )"
,
(
u32
)
(
arm_addr
->
addr_space_buffer
+
offset
),
(
u32
)
req
->
req
.
sendb
,
(
u32
)
req
->
req
.
length
);
if
(
copy_from_user
(
arm_addr
->
addr_space_buffer
+
offset
,
int2ptr
(
req
->
req
.
sendb
),
req
->
req
.
length
))
{
spin_unlock_irqrestore
(
&
host_info_lock
,
flags
);
return
(
-
EFAULT
);
}
spin_unlock_irqrestore
(
&
host_info_lock
,
flags
);
free_pending_request
(
req
);
/* we have to free the request, because we queue no response, and therefore nobody will free it */
return
sizeof
(
struct
raw1394_request
);
}
else
{
DBGMSG
(
"arm_set_buf request exceeded mapping"
);
spin_unlock_irqrestore
(
&
host_info_lock
,
flags
);
return
(
-
EINVAL
);
}
}
entry
=
entry
->
next
;
}
spin_unlock_irqrestore
(
&
host_info_lock
,
flags
);
return
(
-
EINVAL
);
}
static
int
reset_notification
(
struct
file_info
*
fi
,
struct
pending_request
*
req
)
{
DBGMSG
(
"reset_notification called - switch %s "
,
...
...
@@ -1829,9 +1929,8 @@ static int write_phypacket(struct file_info *fi, struct pending_request *req)
spin_unlock_irq
(
&
fi
->
reqlists_lock
);
packet
->
generation
=
req
->
req
.
generation
;
retval
=
hpsb_send_packet
(
packet
);
DBGMSG
(
"write_phypacket send_packet called => retval: %d "
,
retval
);
if
(
!
retval
)
{
DBGMSG
(
"write_phypacket send_packet called => retval: %d "
,
retval
);
if
(
retval
<
0
)
{
req
->
req
.
error
=
RAW1394_ERROR_SEND_ERROR
;
req
->
req
.
length
=
0
;
queue_complete_req
(
req
);
...
...
@@ -1912,6 +2011,12 @@ static int state_connected(struct file_info *fi, struct pending_request *req)
case
RAW1394_REQ_ARM_UNREGISTER
:
return
arm_unregister
(
fi
,
req
);
case
RAW1394_REQ_ARM_SET_BUF
:
return
arm_set_buf
(
fi
,
req
);
case
RAW1394_REQ_ARM_GET_BUF
:
return
arm_get_buf
(
fi
,
req
);
case
RAW1394_REQ_RESET_NOTIFY
:
return
reset_notification
(
fi
,
req
);
...
...
@@ -2137,6 +2242,7 @@ static int raw1394_iso_recv_init(struct file_info *fi, void *uaddr)
stat
.
config
.
data_buf_size
,
stat
.
config
.
buf_packets
,
stat
.
config
.
channel
,
stat
.
config
.
dma_mode
,
stat
.
config
.
irq_interval
,
rawiso_activity_cb
);
if
(
!
fi
->
iso_handle
)
...
...
@@ -2489,7 +2595,7 @@ static int raw1394_release(struct inode *inode, struct file *file)
list_del
(
&
fi
->
list
);
spin_unlock_irq
(
&
host_info_lock
);
hpsb_unref_host
(
fi
->
host
);
put_device
(
&
fi
->
host
->
device
);
}
kfree
(
fi
);
...
...
drivers/ieee1394/raw1394.h
View file @
d911e854
...
...
@@ -30,6 +30,8 @@
#define RAW1394_REQ_ARM_REGISTER 300
#define RAW1394_REQ_ARM_UNREGISTER 301
#define RAW1394_REQ_ARM_SET_BUF 302
#define RAW1394_REQ_ARM_GET_BUF 303
#define RAW1394_REQ_RESET_NOTIFY 400
...
...
@@ -149,6 +151,9 @@ struct raw1394_iso_config {
/* xmit only - iso transmission speed */
__u8
speed
;
/* The mode of the dma when receiving iso data. Must be supported by chip */
__u8
dma_mode
;
/* max. latency of buffer, in packets (-1 if you don't care) */
__s32
irq_interval
;
};
...
...
drivers/ieee1394/sbp2.c
View file @
d911e854
...
...
@@ -77,7 +77,7 @@
#include "sbp2.h"
static
char
version
[]
__devinitdata
=
"$Rev: 10
34
$ Ben Collins <bcollins@debian.org>"
;
"$Rev: 10
82
$ Ben Collins <bcollins@debian.org>"
;
/*
* Module load parameter definitions
...
...
@@ -361,38 +361,34 @@ static int sbp2util_down_timeout(atomic_t *done, int timeout)
static
void
sbp2_free_packet
(
struct
hpsb_packet
*
packet
)
{
hpsb_free_tlabel
(
packet
);
free_hpsb
_packet
(
packet
);
hpsb_free
_packet
(
packet
);
}
/*
* This function is called to retrieve a block write packet from our
* packet pool. This function is used in place of calling
* alloc_hpsb_packet (which costs us three kmallocs). Instead we just pull
* out a free request packet and re-initialize values in it. I'm sure this
* can still stand some more optimization.
/* This is much like hpsb_node_write(), except it ignores the response
* subaction and returns immediately. Can be used from interrupts.
*/
static
struct
hpsb_packet
*
sbp2util_allocate_write_packet
(
struct
sbp2scsi_host_info
*
hi
,
struct
node_entry
*
ne
,
u64
addr
,
size_t
data_size
,
quadlet_t
*
data
)
int
sbp2util_node_write_no_wait
(
struct
node_entry
*
ne
,
u64
addr
,
quadlet_t
*
buffer
,
size_t
length
)
{
struct
hpsb_packet
*
packet
;
packet
=
hpsb_make_writepacket
(
hi
->
host
,
ne
->
nodeid
,
addr
,
data
,
data_size
);
packet
=
hpsb_make_writepacket
(
ne
->
host
,
ne
->
nodeid
,
addr
,
buffer
,
length
);
if
(
!
packet
)
return
NULL
;
return
-
ENOMEM
;
hpsb_set_packet_complete_task
(
packet
,
(
void
(
*
)(
void
*
))
sbp2_free_packet
,
packet
);
hpsb_node_fill_packet
(
ne
,
packet
);
return
packet
;
}
if
(
hpsb_send_packet
(
packet
)
<
0
)
{
sbp2_free_packet
(
packet
);
return
-
EIO
;
}
return
0
;
}
/*
* This function is called to create a pool of command orbs used for
...
...
@@ -1734,35 +1730,26 @@ static int sbp2_max_speed_and_size(struct scsi_id_instance_data *scsi_id)
*/
static
int
sbp2_agent_reset
(
struct
scsi_id_instance_data
*
scsi_id
,
int
wait
)
{
struct
sbp2scsi_host_info
*
hi
=
scsi_id
->
hi
;
struct
hpsb_packet
*
packet
;
quadlet_t
data
;
u64
addr
;
int
retval
;
SBP2_DEBUG
(
"sbp2_agent_reset"
);
/*
* Ok, let's write to the target's management agent register
*/
data
=
ntohl
(
SBP2_AGENT_RESET_DATA
);
packet
=
sbp2util_allocate_write_packet
(
hi
,
scsi_id
->
ne
,
scsi_id
->
sbp2_command_block_agent_addr
+
SBP2_AGENT_RESET_OFFSET
,
4
,
&
data
);
if
(
!
packet
)
{
SBP2_ERR
(
"sbp2util_allocate_write_packet failed"
);
return
(
-
ENOMEM
);
}
addr
=
scsi_id
->
sbp2_command_block_agent_addr
+
SBP2_AGENT_RESET_OFFSET
;
if
(
!
hpsb_send_packet
(
packet
))
{
SBP2_ERR
(
"hpsb_send_packet failed"
);
sbp2_free_packet
(
packet
);
return
(
-
EIO
);
}
if
(
wait
)
retval
=
hpsb_node_write
(
scsi_id
->
ne
,
addr
,
&
data
,
4
);
else
retval
=
sbp2util_node_write_no_wait
(
scsi_id
->
ne
,
addr
,
&
data
,
4
);
if
(
wait
)
{
down
(
&
packet
->
state_change
);
down
(
&
packet
->
state_change
)
;
if
(
retval
<
0
)
{
SBP2_ERR
(
"hpsb_node_write failed.
\n
"
);
return
-
EIO
;
}
/*
...
...
@@ -2032,8 +2019,9 @@ static int sbp2_link_orb_command(struct scsi_id_instance_data *scsi_id,
struct
sbp2_command_info
*
command
)
{
struct
sbp2scsi_host_info
*
hi
=
scsi_id
->
hi
;
struct
hpsb_packet
*
packet
;
struct
sbp2_command_orb
*
command_orb
=
&
command
->
command_orb
;
struct
node_entry
*
ne
=
scsi_id
->
ne
;
u64
addr
;
outstanding_orb_incr
;
SBP2_ORB_DEBUG
(
"sending command orb %p, total orbs = %x"
,
...
...
@@ -2049,40 +2037,30 @@ static int sbp2_link_orb_command(struct scsi_id_instance_data *scsi_id,
* Check to see if there are any previous orbs to use
*/
if
(
scsi_id
->
last_orb
==
NULL
)
{
quadlet_t
data
[
2
];
/*
* Ok, let's write to the target's management agent register
*/
if
(
hpsb_node_entry_valid
(
scsi_id
->
ne
))
{
packet
=
sbp2util_allocate_write_packet
(
hi
,
scsi_id
->
ne
,
scsi_id
->
sbp2_command_block_agent_addr
+
SBP2_ORB_POINTER_OFFSET
,
8
,
NULL
);
if
(
!
packet
)
{
SBP2_ERR
(
"sbp2util_allocate_write_packet failed"
);
return
(
-
ENOMEM
);
}
packet
->
data
[
0
]
=
ORB_SET_NODE_ID
(
hi
->
host
->
node_id
);
packet
->
data
[
1
]
=
command
->
command_orb_dma
;
sbp2util_cpu_to_be32_buffer
(
packet
->
data
,
8
);
SBP2_ORB_DEBUG
(
"write command agent, command orb %p"
,
command_orb
);
addr
=
scsi_id
->
sbp2_command_block_agent_addr
+
SBP2_ORB_POINTER_OFFSET
;
data
[
0
]
=
ORB_SET_NODE_ID
(
hi
->
host
->
node_id
);
data
[
1
]
=
command
->
command_orb_dma
;
sbp2util_cpu_to_be32_buffer
(
data
,
8
);
if
(
!
hpsb_send_packet
(
packet
))
{
SBP2_ERR
(
"hpsb_send_packet failed"
);
sbp2_free_packet
(
packet
);
return
(
-
EIO
);
}
SBP2_ORB_DEBUG
(
"write command agent, command orb %p"
,
command_orb
);
SBP2_ORB_DEBUG
(
"write command agent complete"
);
if
(
sbp2util_node_write_no_wait
(
ne
,
addr
,
data
,
8
)
<
0
)
{
SBP2_ERR
(
"sbp2util_node_write_no_wait failed.
\n
"
);
return
-
EIO
;
}
SBP2_ORB_DEBUG
(
"write command agent complete"
);
scsi_id
->
last_orb
=
command_orb
;
scsi_id
->
last_orb_dma
=
command
->
command_orb_dma
;
}
else
{
quadlet_t
data
;
/*
* We have an orb already sent (maybe or maybe not
...
...
@@ -2102,25 +2080,14 @@ static int sbp2_link_orb_command(struct scsi_id_instance_data *scsi_id,
/*
* Ring the doorbell
*/
if
(
hpsb_node_entry_valid
(
scsi_id
->
ne
))
{
quadlet_t
data
=
cpu_to_be32
(
command
->
command_orb_dma
)
;
data
=
cpu_to_be32
(
command
->
command_orb_dma
);
addr
=
scsi_id
->
sbp2_command_block_agent_addr
+
SBP2_DOORBELL_OFFSET
;
packet
=
sbp2util_allocate_write_packet
(
hi
,
scsi_id
->
ne
,
scsi_id
->
sbp2_command_block_agent_addr
+
SBP2_DOORBELL_OFFSET
,
4
,
&
data
);
if
(
!
packet
)
{
SBP2_ERR
(
"sbp2util_allocate_write_packet failed"
);
return
(
-
ENOMEM
);
}
SBP2_ORB_DEBUG
(
"ring doorbell, command orb %p"
,
command_orb
);
SBP2_ORB_DEBUG
(
"ring doorbell, command orb %p"
,
command_orb
);
if
(
!
hpsb_send_packet
(
packet
))
{
SBP2_ERR
(
"hpsb_send_packet failed"
);
sbp2_free_packet
(
packet
);
return
(
-
EIO
);
}
if
(
sbp2util_node_write_no_wait
(
ne
,
addr
,
&
data
,
4
)
<
0
)
{
SBP2_ERR
(
"sbp2util_node_write_no_wait failed"
);
return
(
-
EIO
);
}
scsi_id
->
last_orb
=
command_orb
;
...
...
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