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
40d1b71b
Commit
40d1b71b
authored
Oct 15, 2002
by
Linus Torvalds
Browse files
Options
Browse Files
Download
Plain Diff
Merge
bk://linux-bt.bkbits.net/bt-2.5
into home.transmeta.com:/home/torvalds/v2.5/linux
parents
4c9eb495
0c8aab5d
Changes
8
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
481 additions
and
233 deletions
+481
-233
drivers/block/Makefile
drivers/block/Makefile
+2
-2
drivers/block/blkpg.c
drivers/block/blkpg.c
+1
-1
drivers/block/scsi_ioctl.c
drivers/block/scsi_ioctl.c
+215
-0
drivers/ide/ide-cd.c
drivers/ide/ide-cd.c
+248
-183
drivers/ide/ide-cd.h
drivers/ide/ide-cd.h
+0
-7
drivers/ide/ide.c
drivers/ide/ide.c
+1
-1
fs/block_dev.c
fs/block_dev.c
+3
-38
include/linux/blkdev.h
include/linux/blkdev.h
+11
-1
No files found.
drivers/block/Makefile
View file @
40d1b71b
...
...
@@ -9,9 +9,9 @@
#
export-objs
:=
elevator.o ll_rw_blk.o loop.o genhd.o acsi.o
\
block
_ioctl.o deadline-iosched.o
scsi
_ioctl.o deadline-iosched.o
obj-y
:=
elevator.o ll_rw_blk.o blkpg.o genhd.o
block
_ioctl.o deadline-iosched.o
obj-y
:=
elevator.o ll_rw_blk.o blkpg.o genhd.o
scsi
_ioctl.o deadline-iosched.o
obj-$(CONFIG_MAC_FLOPPY)
+=
swim3.o
obj-$(CONFIG_BLK_DEV_FD)
+=
floppy.o
...
...
drivers/block/blkpg.c
View file @
40d1b71b
...
...
@@ -305,6 +305,6 @@ int blk_ioctl(struct block_device *bdev, unsigned int cmd, unsigned long arg)
return
0
;
default:
return
-
E
INVAL
;
return
-
E
NOTTY
;
}
}
drivers/block/
block
_ioctl.c
→
drivers/block/
scsi
_ioctl.c
View file @
40d1b71b
...
...
@@ -27,8 +27,12 @@
#include <linux/module.h>
#include <linux/blk.h>
#include <linux/completion.h>
#include <linux/cdrom.h>
#include <linux/slab.h>
#include <scsi/scsi.h>
#include <asm/uaccess.h>
int
blk_do_rq
(
request_queue_t
*
q
,
struct
request
*
rq
)
{
...
...
@@ -50,7 +54,114 @@ int blk_do_rq(request_queue_t *q, struct request *rq)
return
err
;
}
int
block_ioctl
(
struct
block_device
*
bdev
,
unsigned
int
cmd
,
unsigned
long
arg
)
#include <scsi/sg.h>
static
int
sg_get_version
(
int
*
p
)
{
static
int
sg_version_num
=
30527
;
return
put_user
(
sg_version_num
,
p
);
}
static
int
scsi_get_idlun
(
request_queue_t
*
q
,
int
*
p
)
{
return
put_user
(
0
,
p
);
}
static
int
scsi_get_bus
(
request_queue_t
*
q
,
int
*
p
)
{
return
put_user
(
0
,
p
);
}
static
int
sg_get_timeout
(
request_queue_t
*
q
)
{
return
HZ
;
}
static
int
sg_set_timeout
(
request_queue_t
*
q
,
int
*
p
)
{
int
timeout
;
int
error
=
get_user
(
timeout
,
p
);
return
error
;
}
static
int
reserved_size
=
0
;
static
int
sg_get_reserved_size
(
request_queue_t
*
q
,
int
*
p
)
{
return
put_user
(
reserved_size
,
p
);
}
static
int
sg_set_reserved_size
(
request_queue_t
*
q
,
int
*
p
)
{
int
size
;
int
error
=
get_user
(
size
,
p
);
if
(
!
error
)
reserved_size
=
size
;
return
error
;
}
static
int
sg_emulated_host
(
request_queue_t
*
q
,
int
*
p
)
{
return
put_user
(
1
,
p
);
}
static
int
sg_io
(
request_queue_t
*
q
,
struct
sg_io_hdr
*
uptr
)
{
int
err
;
struct
sg_io_hdr
hdr
;
struct
request
*
rq
;
void
*
buffer
;
if
(
!
access_ok
(
VERIFY_WRITE
,
uptr
,
sizeof
(
*
uptr
)))
return
-
EFAULT
;
if
(
copy_from_user
(
&
hdr
,
uptr
,
sizeof
(
*
uptr
)))
return
-
EFAULT
;
if
(
hdr
.
cmd_len
>
sizeof
(
rq
->
cmd
)
)
return
-
EINVAL
;
buffer
=
NULL
;
if
(
hdr
.
dxfer_len
)
{
unsigned
int
bytes
=
(
hdr
.
dxfer_len
+
511
)
&
~
511
;
switch
(
hdr
.
dxfer_direction
)
{
default:
return
-
EINVAL
;
case
SG_DXFER_TO_DEV
:
case
SG_DXFER_FROM_DEV
:
case
SG_DXFER_TO_FROM_DEV
:
break
;
}
buffer
=
kmalloc
(
bytes
,
GFP_USER
);
if
(
!
buffer
)
return
-
ENOMEM
;
if
(
hdr
.
dxfer_direction
==
SG_DXFER_TO_DEV
||
hdr
.
dxfer_direction
==
SG_DXFER_TO_FROM_DEV
)
copy_from_user
(
buffer
,
hdr
.
dxferp
,
hdr
.
dxfer_len
);
}
rq
=
blk_get_request
(
q
,
WRITE
,
__GFP_WAIT
);
rq
->
timeout
=
60
*
HZ
;
rq
->
data
=
buffer
;
rq
->
data_len
=
hdr
.
dxfer_len
;
rq
->
flags
=
REQ_BLOCK_PC
;
memset
(
rq
->
cmd
,
0
,
sizeof
(
rq
->
cmd
));
copy_from_user
(
rq
->
cmd
,
hdr
.
cmdp
,
hdr
.
cmd_len
);
err
=
blk_do_rq
(
q
,
rq
);
blk_put_request
(
rq
);
copy_to_user
(
uptr
,
&
hdr
,
sizeof
(
*
uptr
));
if
(
buffer
)
{
if
(
hdr
.
dxfer_direction
==
SG_DXFER_FROM_DEV
||
hdr
.
dxfer_direction
==
SG_DXFER_TO_FROM_DEV
)
copy_to_user
(
hdr
.
dxferp
,
buffer
,
hdr
.
dxfer_len
);
kfree
(
buffer
);
}
return
err
;
}
int
scsi_cmd_ioctl
(
struct
block_device
*
bdev
,
unsigned
int
cmd
,
unsigned
long
arg
)
{
request_queue_t
*
q
;
struct
request
*
rq
;
...
...
@@ -61,11 +172,32 @@ int block_ioctl(struct block_device *bdev, unsigned int cmd, unsigned long arg)
return
-
ENXIO
;
switch
(
cmd
)
{
case
SG_GET_VERSION_NUM
:
return
sg_get_version
((
int
*
)
arg
);
case
SCSI_IOCTL_GET_IDLUN
:
return
scsi_get_idlun
(
q
,
(
int
*
)
arg
);
case
SCSI_IOCTL_GET_BUS_NUMBER
:
return
scsi_get_bus
(
q
,
(
int
*
)
arg
);
case
SG_SET_TIMEOUT
:
return
sg_set_timeout
(
q
,
(
int
*
)
arg
);
case
SG_GET_TIMEOUT
:
return
sg_get_timeout
(
q
);
case
SG_GET_RESERVED_SIZE
:
return
sg_get_reserved_size
(
q
,
(
int
*
)
arg
);
case
SG_SET_RESERVED_SIZE
:
return
sg_set_reserved_size
(
q
,
(
int
*
)
arg
);
case
SG_EMULATED_HOST
:
return
sg_emulated_host
(
q
,
(
int
*
)
arg
);
case
SG_IO
:
return
sg_io
(
q
,
(
struct
sg_io_hdr
*
)
arg
);
case
CDROMCLOSETRAY
:
close
=
1
;
case
CDROMEJECT
:
rq
=
blk_get_request
(
q
,
WRITE
,
__GFP_WAIT
);
rq
->
flags
=
REQ_BLOCK_PC
;
rq
->
data
=
NULL
;
rq
->
data_len
=
0
;
rq
->
timeout
=
60
*
HZ
;
memset
(
rq
->
cmd
,
0
,
sizeof
(
rq
->
cmd
));
rq
->
cmd
[
0
]
=
GPCMD_START_STOP_UNIT
;
rq
->
cmd
[
4
]
=
0x02
+
(
close
!=
0
);
...
...
@@ -80,4 +212,4 @@ int block_ioctl(struct block_device *bdev, unsigned int cmd, unsigned long arg)
return
err
;
}
EXPORT_SYMBOL
(
block
_ioctl
);
EXPORT_SYMBOL
(
scsi_cmd
_ioctl
);
drivers/ide/ide-cd.c
View file @
40d1b71b
...
...
@@ -309,6 +309,8 @@
#include <linux/ide.h>
#include <linux/completion.h>
#include <scsi/scsi.h>
/* For SCSI -> ATAPI command conversion */
#include <asm/irq.h>
#include <asm/io.h>
#include <asm/byteorder.h>
...
...
@@ -332,12 +334,12 @@ static void cdrom_saw_media_change (ide_drive_t *drive)
info
->
nsectors_buffered
=
0
;
}
static
int
cdrom_log_sense
(
ide_drive_t
*
drive
,
struct
packet_command
*
pc
,
static
int
cdrom_log_sense
(
ide_drive_t
*
drive
,
struct
request
*
rq
,
struct
request_sense
*
sense
)
{
int
log
=
0
;
if
(
sense
==
NULL
||
pc
==
NULL
||
pc
->
quiet
)
if
(
!
sense
||
!
rq
||
(
rq
->
flags
&
REQ_QUIET
)
)
return
0
;
switch
(
sense
->
sense_key
)
{
...
...
@@ -370,10 +372,9 @@ static int cdrom_log_sense(ide_drive_t *drive, struct packet_command *pc,
static
void
cdrom_analyze_sense_data
(
ide_drive_t
*
drive
,
struct
packet_command
*
failed_command
,
struct
request
*
failed_command
,
struct
request_sense
*
sense
)
{
if
(
!
cdrom_log_sense
(
drive
,
failed_command
,
sense
))
return
;
...
...
@@ -382,7 +383,7 @@ void cdrom_analyze_sense_data(ide_drive_t *drive,
* the first toc has not been recorded yet, it will fail with
* 05/24/00 (which is a confusing error)
*/
if
(
failed_command
&&
failed_command
->
c
[
0
]
==
GPCMD_READ_TOC_PMA_ATIP
)
if
(
failed_command
&&
failed_command
->
c
md
[
0
]
==
GPCMD_READ_TOC_PMA_ATIP
)
if
(
sense
->
sense_key
==
0x05
&&
sense
->
asc
==
0x24
)
return
;
...
...
@@ -453,20 +454,20 @@ void cdrom_analyze_sense_data(ide_drive_t *drive,
while
(
hi
>
lo
)
{
mid
=
(
lo
+
hi
)
/
2
;
if
(
packet_command_texts
[
mid
].
packet_command
==
failed_command
->
c
[
0
])
{
failed_command
->
c
md
[
0
])
{
s
=
packet_command_texts
[
mid
].
text
;
break
;
}
if
(
packet_command_texts
[
mid
].
packet_command
>
failed_command
->
c
[
0
])
failed_command
->
c
md
[
0
])
hi
=
mid
;
else
lo
=
mid
+
1
;
}
printk
(
" The failed
\"
%s
\"
packet command was:
\n
\"
"
,
s
);
for
(
i
=
0
;
i
<
sizeof
(
failed_command
->
c
);
i
++
)
printk
(
"%02x "
,
failed_command
->
c
[
i
]);
for
(
i
=
0
;
i
<
sizeof
(
failed_command
->
c
md
);
i
++
)
printk
(
"%02x "
,
failed_command
->
c
md
[
i
]);
printk
(
"
\"\n
"
);
}
...
...
@@ -512,30 +513,39 @@ void cdrom_analyze_sense_data(ide_drive_t *drive,
#endif
/* not VERBOSE_IDE_CD_ERRORS */
}
/*
* Initialize a ide-cd packet command request
*/
static
void
cdrom_prepare_request
(
struct
request
*
rq
)
{
ide_init_drive_cmd
(
rq
);
rq
->
flags
=
REQ_PC
;
}
static
void
cdrom_queue_request_sense
(
ide_drive_t
*
drive
,
struct
completion
*
wait
,
struct
request_sense
*
sense
,
struct
packet_command
*
failed_command
)
void
*
sense
,
struct
request
*
failed_command
)
{
struct
cdrom_info
*
info
=
drive
->
driver_data
;
struct
packet_command
*
pc
=
&
info
->
request_sense_pc
;
struct
request
*
rq
;
struct
request
*
rq
=
&
info
->
request_sense_request
;
if
(
sense
==
NULL
)
sense
=
&
info
->
sense_data
;
memset
(
pc
,
0
,
sizeof
(
struct
packet_command
));
pc
->
c
[
0
]
=
GPCMD_REQUEST_SENSE
;
pc
->
c
[
4
]
=
pc
->
buflen
=
18
;
pc
->
buffer
=
(
char
*
)
sense
;
pc
->
sense
=
(
struct
request_sense
*
)
failed_command
;
/* stuff the sense request in front of our current request */
rq
=
&
info
->
request_sense_request
;
ide_init_drive_cmd
(
rq
);
cdrom_prepare_request
(
rq
);
rq
->
data
=
sense
;
rq
->
cmd
[
0
]
=
GPCMD_REQUEST_SENSE
;
rq
->
cmd
[
4
]
=
rq
->
data_len
=
18
;
rq
->
flags
=
REQ_SENSE
;
rq
->
buffer
=
(
char
*
)
pc
;
rq
->
waiting
=
wait
;
/* NOTE! Save the failed command in "rq->buffer" */
rq
->
buffer
=
(
void
*
)
failed_command
;
(
void
)
ide_do_drive_cmd
(
drive
,
rq
,
ide_preempt
);
}
...
...
@@ -630,17 +640,26 @@ static void cdrom_end_request (ide_drive_t *drive, int uptodate)
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
if
((
rq
->
flags
&
REQ_SENSE
)
&&
uptodate
)
{
struct
packet_command
*
pc
=
(
struct
packet_command
*
)
rq
->
buffer
;
cdrom_analyze_sense_data
(
drive
,
(
struct
packet_command
*
)
pc
->
sense
,
(
struct
request_sense
*
)
(
pc
->
buffer
-
pc
->
c
[
4
]));
/* For REQ_SENSE, "rq->buffer" points to the original failed request */
struct
request
*
failed
=
(
struct
request
*
)
rq
->
buffer
;
struct
cdrom_info
*
info
=
drive
->
driver_data
;
void
*
sense
=
&
info
->
sense_data
;
if
(
failed
&&
failed
->
sense
)
sense
=
failed
->
sense
;
cdrom_analyze_sense_data
(
drive
,
failed
,
sense
);
}
if
(
blk_fs_request
(
rq
)
&&
!
rq
->
current_nr_sectors
)
uptodate
=
1
;
ide_end_request
(
drive
,
uptodate
,
rq
->
hard_cur_sectors
);
}
/* Handle differences between SCSI and ATAPI packet commands */
static
int
pre_transform_command
(
struct
request
*
);
static
void
post_transform_command
(
struct
request
*
);
/* Returns 0 if the request should be continued.
Returns 1 if the request was ended. */
...
...
@@ -649,7 +668,6 @@ static int cdrom_decode_status (ide_startstop_t *startstop, ide_drive_t *drive,
{
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
int
stat
,
err
,
sense_key
;
struct
packet_command
*
pc
;
/* Check for errors. */
*
stat_ret
=
stat
=
HWIF
(
drive
)
->
INB
(
IDE_STATUS_REG
);
...
...
@@ -672,16 +690,18 @@ static int cdrom_decode_status (ide_startstop_t *startstop, ide_drive_t *drive,
from the drive (probably while trying
to recover from a former error). Just give up. */
pc
=
(
struct
packet_command
*
)
rq
->
buffer
;
pc
->
stat
=
1
;
cdrom_end_request
(
drive
,
1
);
rq
->
flags
|=
REQ_FAILED
;
cdrom_end_request
(
drive
,
0
);
*
startstop
=
DRIVER
(
drive
)
->
error
(
drive
,
"request sense failure"
,
stat
);
return
1
;
}
else
if
(
rq
->
flags
&
(
REQ_PC
|
REQ_BLOCK_PC
))
{
/* All other functions, except for READ. */
struct
completion
*
wait
=
NULL
;
pc
=
(
struct
packet_command
*
)
rq
->
buffer
;
/* Fix up any SCSI command differences.. */
if
(
rq
->
flags
&
REQ_BLOCK_PC
)
post_transform_command
(
rq
);
/* Check for tray open. */
if
(
sense_key
==
NOT_READY
)
{
...
...
@@ -691,7 +711,7 @@ static int cdrom_decode_status (ide_startstop_t *startstop, ide_drive_t *drive,
cdrom_saw_media_change
(
drive
);
/*printk("%s: media changed\n",drive->name);*/
return
0
;
}
else
if
(
!
pc
->
quiet
)
{
}
else
if
(
!
(
rq
->
flags
&
REQ_QUIET
)
)
{
/* Otherwise, print an error. */
ide_dump_status
(
drive
,
"packet command error"
,
stat
);
}
...
...
@@ -710,11 +730,11 @@ static int cdrom_decode_status (ide_startstop_t *startstop, ide_drive_t *drive,
rq
->
waiting
=
NULL
;
}
pc
->
stat
=
1
;
cdrom_end_request
(
drive
,
1
);
rq
->
flags
|=
REQ_FAILED
;
cdrom_end_request
(
drive
,
0
);
if
((
stat
&
ERR_STAT
)
!=
0
)
cdrom_queue_request_sense
(
drive
,
wait
,
pc
->
sense
,
pc
);
cdrom_queue_request_sense
(
drive
,
wait
,
rq
->
sense
,
rq
);
}
else
if
(
blk_fs_request
(
rq
))
{
/* Handle errors from READ and WRITE requests. */
...
...
@@ -770,7 +790,6 @@ static int cdrom_decode_status (ide_startstop_t *startstop, ide_drive_t *drive,
static
int
cdrom_timer_expiry
(
ide_drive_t
*
drive
)
{
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
struct
packet_command
*
pc
=
(
struct
packet_command
*
)
rq
->
buffer
;
unsigned
long
wait
=
0
;
/*
...
...
@@ -779,7 +798,7 @@ static int cdrom_timer_expiry(ide_drive_t *drive)
* this, but not all commands/drives support that. Let
* ide_timer_expiry keep polling us for these.
*/
switch
(
pc
->
c
[
0
])
{
switch
(
rq
->
cmd
[
0
])
{
case
GPCMD_BLANK
:
case
GPCMD_FORMAT_UNIT
:
case
GPCMD_RESERVE_RZONE_TRACK
:
...
...
@@ -854,12 +873,12 @@ static ide_startstop_t cdrom_start_packet_command(ide_drive_t *drive,
* struct packet_command *pc; now packet_command_t *pc;
*/
static
ide_startstop_t
cdrom_transfer_packet_command
(
ide_drive_t
*
drive
,
struct
packet_command
*
pc
,
struct
request
*
rq
,
ide_handler_t
*
handler
)
{
unsigned
char
*
cmd_buf
=
pc
->
c
;
int
cmd_len
=
sizeof
(
pc
->
c
);
unsigned
int
timeout
=
pc
->
timeout
;
unsigned
char
*
cmd_buf
=
rq
->
cmd
;
int
cmd_len
=
sizeof
(
rq
->
cmd
);
unsigned
int
timeout
=
rq
->
timeout
;
struct
cdrom_info
*
info
=
drive
->
driver_data
;
ide_startstop_t
startstop
;
...
...
@@ -1029,6 +1048,7 @@ static ide_startstop_t cdrom_read_intr (ide_drive_t *drive)
if
(
rq
->
current_nr_sectors
>
0
)
{
printk
(
"%s: cdrom_read_intr: data underrun (%d blocks)
\n
"
,
drive
->
name
,
rq
->
current_nr_sectors
);
rq
->
flags
|=
REQ_FAILED
;
cdrom_end_request
(
drive
,
0
);
}
else
cdrom_end_request
(
drive
,
1
);
...
...
@@ -1179,7 +1199,6 @@ static int cdrom_read_from_buffer (ide_drive_t *drive)
*/
static
ide_startstop_t
cdrom_start_read_continuation
(
ide_drive_t
*
drive
)
{
struct
packet_command
pc
;
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
int
nsect
,
sector
,
nframes
,
frame
,
nskip
;
...
...
@@ -1222,11 +1241,10 @@ static ide_startstop_t cdrom_start_read_continuation (ide_drive_t *drive)
(
65534
/
CD_FRAMESIZE
)
:
65535
);
/* Set up the command */
memcpy
(
pc
.
c
,
rq
->
cmd
,
sizeof
(
pc
.
c
));
pc
.
timeout
=
WAIT_CMD
;
rq
->
timeout
=
WAIT_CMD
;
/* Send the command to the drive and return. */
return
cdrom_transfer_packet_command
(
drive
,
&
pc
,
&
cdrom_read_intr
);
return
cdrom_transfer_packet_command
(
drive
,
rq
,
&
cdrom_read_intr
);
}
...
...
@@ -1262,7 +1280,6 @@ static ide_startstop_t cdrom_seek_intr (ide_drive_t *drive)
static
ide_startstop_t
cdrom_start_seek_continuation
(
ide_drive_t
*
drive
)
{
struct
packet_command
pc
;
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
int
sector
,
frame
,
nskip
;
...
...
@@ -1273,11 +1290,11 @@ static ide_startstop_t cdrom_start_seek_continuation (ide_drive_t *drive)
frame
=
sector
/
SECTORS_PER_FRAME
;
memset
(
rq
->
cmd
,
0
,
sizeof
(
rq
->
cmd
));
pc
.
c
[
0
]
=
GPCMD_SEEK
;
put_unaligned
(
cpu_to_be32
(
frame
),
(
unsigned
int
*
)
&
pc
.
c
[
2
]);
rq
->
cmd
[
0
]
=
GPCMD_SEEK
;
put_unaligned
(
cpu_to_be32
(
frame
),
(
unsigned
int
*
)
&
rq
->
cmd
[
2
]);
pc
.
timeout
=
WAIT_CMD
;
return
cdrom_transfer_packet_command
(
drive
,
&
pc
,
&
cdrom_seek_intr
);
rq
->
timeout
=
WAIT_CMD
;
return
cdrom_transfer_packet_command
(
drive
,
rq
,
&
cdrom_seek_intr
);
}
static
ide_startstop_t
cdrom_start_seek
(
ide_drive_t
*
drive
,
unsigned
int
block
)
...
...
@@ -1296,6 +1313,7 @@ static void restore_request (struct request *rq)
{
if
(
rq
->
buffer
!=
bio_data
(
rq
->
bio
))
{
sector_t
n
=
(
rq
->
buffer
-
(
char
*
)
bio_data
(
rq
->
bio
))
/
SECTOR_SIZE
;
rq
->
buffer
=
bio_data
(
rq
->
bio
);
rq
->
nr_sectors
+=
n
;
rq
->
sector
-=
n
;
...
...
@@ -1352,7 +1370,6 @@ static ide_startstop_t cdrom_pc_intr (ide_drive_t *drive)
{
int
ireason
,
len
,
stat
,
thislen
;
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
struct
packet_command
*
pc
=
(
struct
packet_command
*
)
rq
->
buffer
;
ide_startstop_t
startstop
;
u8
lowcyl
=
0
,
highcyl
=
0
;
...
...
@@ -1372,16 +1389,16 @@ static ide_startstop_t cdrom_pc_intr (ide_drive_t *drive)
if
((
stat
&
DRQ_STAT
)
==
0
)
{
/* Some of the trailing request sense fields are optional, and
some drives don't send them. Sigh. */
if
(
pc
->
c
[
0
]
==
GPCMD_REQUEST_SENSE
&&
pc
->
buf
len
>
0
&&
pc
->
buf
len
<=
5
)
{
while
(
pc
->
buf
len
>
0
)
{
*
pc
->
buffer
++
=
0
;
--
pc
->
buf
len
;
if
(
rq
->
cmd
[
0
]
==
GPCMD_REQUEST_SENSE
&&
rq
->
data_
len
>
0
&&
rq
->
data_
len
<=
5
)
{
while
(
rq
->
data_
len
>
0
)
{
*
(
unsigned
char
*
)
rq
->
data
++
=
0
;
--
rq
->
data_
len
;
}
}
if
(
pc
->
buf
len
==
0
)
if
(
rq
->
data_
len
==
0
)
cdrom_end_request
(
drive
,
1
);
else
{
/* Comment this out, because this always happens
...
...
@@ -1391,20 +1408,22 @@ static ide_startstop_t cdrom_pc_intr (ide_drive_t *drive)
printk ("%s: cdrom_pc_intr: data underrun %d\n",
drive->name, pc->buflen);
*/
pc
->
stat
=
1
;
cdrom_end_request
(
drive
,
1
);
rq
->
flags
|=
REQ_FAILED
;
cdrom_end_request
(
drive
,
0
);
}
return
ide_stopped
;
}
/* Figure out how much data to transfer. */
thislen
=
pc
->
buf
len
;
thislen
=
rq
->
data_
len
;
if
(
thislen
>
len
)
thislen
=
len
;
/* The drive wants to be written to. */
if
((
ireason
&
3
)
==
0
)
{
if
(
!
rq
->
data
)
goto
confused
;
/* Transfer the data. */
HWIF
(
drive
)
->
atapi_output_bytes
(
drive
,
pc
->
buffer
,
thislen
);
HWIF
(
drive
)
->
atapi_output_bytes
(
drive
,
rq
->
data
,
thislen
);
/* If we haven't moved enough data to satisfy the drive,
add some padding. */
...
...
@@ -1415,15 +1434,16 @@ static ide_startstop_t cdrom_pc_intr (ide_drive_t *drive)
}
/* Keep count of how much data we've moved. */
pc
->
buffer
+=
thislen
;
pc
->
buf
len
-=
thislen
;
rq
->
data
+=
thislen
;
rq
->
data_
len
-=
thislen
;
}
/* Same drill for reading. */
else
if
((
ireason
&
3
)
==
2
)
{
if
(
!
rq
->
data
)
goto
confused
;
/* Transfer the data. */
HWIF
(
drive
)
->
atapi_input_bytes
(
drive
,
pc
->
buffer
,
thislen
);
HWIF
(
drive
)
->
atapi_input_bytes
(
drive
,
rq
->
data
,
thislen
);
/* If we haven't moved enough data to satisfy the drive,
add some padding. */
...
...
@@ -1434,13 +1454,14 @@ static ide_startstop_t cdrom_pc_intr (ide_drive_t *drive)
}
/* Keep count of how much data we've moved. */
pc
->
buffer
+=
thislen
;
pc
->
buf
len
-=
thislen
;
rq
->
data
+=
thislen
;
rq
->
data_
len
-=
thislen
;
}
else
{
confused:
printk
(
"%s: cdrom_pc_intr: The drive "
"appears confused (ireason = 0x%2x)
\n
"
,
drive
->
name
,
ireason
);
pc
->
stat
=
1
;
rq
->
flags
|=
REQ_FAILED
;
}
if
(
HWGROUP
(
drive
)
->
handler
!=
NULL
)
...
...
@@ -1455,13 +1476,12 @@ static ide_startstop_t cdrom_pc_intr (ide_drive_t *drive)
static
ide_startstop_t
cdrom_do_pc_continuation
(
ide_drive_t
*
drive
)
{
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
struct
packet_command
*
pc
=
(
struct
packet_command
*
)
rq
->
buffer
;
if
(
!
pc
->
timeout
)
pc
->
timeout
=
WAIT_CMD
;
if
(
!
rq
->
timeout
)
rq
->
timeout
=
WAIT_CMD
;
/* Send the command to the drive and return. */
return
cdrom_transfer_packet_command
(
drive
,
pc
,
&
cdrom_pc_intr
);
return
cdrom_transfer_packet_command
(
drive
,
rq
,
&
cdrom_pc_intr
);
}
...
...
@@ -1469,13 +1489,12 @@ static ide_startstop_t cdrom_do_packet_command (ide_drive_t *drive)
{
int
len
;
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
struct
packet_command
*
pc
=
(
struct
packet_command
*
)
rq
->
buffer
;
struct
cdrom_info
*
info
=
drive
->
driver_data
;
info
->
dma
=
0
;
info
->
cmd
=
0
;
pc
->
stat
=
0
;
len
=
pc
->
buf
len
;
rq
->
flags
&=
~
REQ_FAILED
;
len
=
rq
->
data_
len
;
/* Start sending the command to the drive. */
return
cdrom_start_packet_command
(
drive
,
len
,
cdrom_do_pc_continuation
);
...
...
@@ -1496,28 +1515,31 @@ void cdrom_sleep (int time)
}
static
int
cdrom_queue_packet_command
(
ide_drive_t
*
drive
,
struct
packet_command
*
pc
)
int
cdrom_queue_packet_command
(
ide_drive_t
*
drive
,
struct
request
*
rq
)
{
struct
request_sense
sense
;
struct
request
req
;
int
retries
=
10
;
unsigned
int
flags
=
rq
->
flags
;
if
(
pc
->
sense
==
NULL
)
pc
->
sense
=
&
sense
;
if
(
rq
->
sense
==
NULL
)
rq
->
sense
=
&
sense
;
/* Start of retry loop. */
do
{
ide_init_drive_cmd
(
&
req
);
req
.
flags
=
REQ_PC
;
req
.
buffer
=
(
char
*
)
pc
;
ide_do_drive_cmd
(
drive
,
&
req
,
ide_wait
);
int
error
;
unsigned
long
time
=
jiffies
;
rq
->
flags
=
flags
;
error
=
ide_do_drive_cmd
(
drive
,
rq
,
ide_wait
);
time
=
jiffies
-
time
;
/* FIXME: we should probably abort/retry or something
* in case of failure */
if
(
pc
->
stat
!=
0
)
{
if
(
rq
->
flags
&
REQ_FAILED
)
{
/* The request failed. Retry if it was due to a unit
attention status
(usually means media was changed). */
struct
request_sense
*
reqbuf
=
pc
->
sense
;
struct
request_sense
*
reqbuf
=
rq
->
sense
;
if
(
reqbuf
->
sense_key
==
UNIT_ATTENTION
)
cdrom_saw_media_change
(
drive
);
...
...
@@ -1535,10 +1557,10 @@ int cdrom_queue_packet_command(ide_drive_t *drive, struct packet_command *pc)
}
/* End of retry loop. */
}
while
(
pc
->
stat
!=
0
&&
retries
>=
0
);
}
while
(
(
rq
->
flags
&
REQ_FAILED
)
&&
retries
>=
0
);
/* Return an error if the command failed. */
return
pc
->
stat
?
-
EIO
:
0
;
return
(
rq
->
flags
&
REQ_FAILED
)
?
-
EIO
:
0
;
}
/*
...
...
@@ -1681,20 +1703,18 @@ static ide_startstop_t cdrom_write_intr(ide_drive_t *drive)
static
ide_startstop_t
cdrom_start_write_cont
(
ide_drive_t
*
drive
)
{
struct
packet_command
pc
;
/* packet_command_t pc; */
struct
request
*
rq
=
HWGROUP
(
drive
)
->
rq
;
unsigned
nframes
,
frame
;
nframes
=
rq
->
nr_sectors
>>
2
;
frame
=
rq
->
sector
>>
2
;
memcpy
(
pc
.
c
,
rq
->
cmd
,
sizeof
(
pc
.
c
));
#if 0 /* the immediate bit */
pc.c
[1] = 1 << 3;
rq->cmd
[1] = 1 << 3;
#endif
pc
.
timeout
=
2
*
WAIT_CMD
;
rq
->
timeout
=
2
*
WAIT_CMD
;
return
cdrom_transfer_packet_command
(
drive
,
&
pc
,
cdrom_write_intr
);
return
cdrom_transfer_packet_command
(
drive
,
rq
,
cdrom_write_intr
);
}
static
ide_startstop_t
cdrom_start_write
(
ide_drive_t
*
drive
,
struct
request
*
rq
)
...
...
@@ -1728,20 +1748,54 @@ static ide_startstop_t cdrom_start_write(ide_drive_t *drive, struct request *rq)
return
cdrom_start_packet_command
(
drive
,
32768
,
cdrom_start_write_cont
);
}
/*
* Most of the SCSI commands are supported directly by ATAPI devices.
* This transform handles the few exceptions.
*/
static
int
pre_transform_command
(
struct
request
*
req
)
{
u8
*
c
=
req
->
cmd
;
/* Transform 6-byte read/write commands to the 10-byte version. */
if
(
c
[
0
]
==
READ_6
||
c
[
0
]
==
WRITE_6
)
{
c
[
8
]
=
c
[
4
];
c
[
5
]
=
c
[
3
];
c
[
4
]
=
c
[
2
];
c
[
3
]
=
c
[
1
]
&
0x1f
;
c
[
2
]
=
0
;
c
[
1
]
&=
0xe0
;
c
[
0
]
+=
(
READ_10
-
READ_6
);
return
0
;
}
/* These also need fixup, not done yet */
if
(
c
[
0
]
==
MODE_SENSE
||
c
[
0
]
==
MODE_SELECT
)
return
-
EINVAL
;
return
0
;
}
static
void
post_transform_command
(
struct
request
*
req
)
{
}
static
ide_startstop_t
cdrom_do_block_pc
(
ide_drive_t
*
drive
,
struct
request
*
rq
)
{
struct
packet_command
pc
;
ide_startstop_t
startstop
;
struct
cdrom_info
*
info
;
memset
(
&
pc
,
0
,
sizeof
(
pc
));
memcpy
(
pc
.
c
,
rq
->
cmd
,
sizeof
(
pc
.
c
));
pc
.
quiet
=
1
;
pc
.
timeout
=
60
*
HZ
;
rq
->
buffer
=
(
char
*
)
&
pc
;
if
(
pre_transform_command
(
rq
)
<
0
)
{
cdrom_end_request
(
drive
,
0
);
return
ide_stopped
;
}
rq
->
flags
|=
REQ_QUIET
;
startstop
=
cdrom_do_packet_command
(
drive
);
if
(
pc
.
stat
)
rq
->
errors
++
;
info
=
drive
->
driver_data
;
info
->
dma
=
0
;
info
->
cmd
=
0
;
/* Start sending the command to the drive. */
startstop
=
cdrom_start_packet_command
(
drive
,
rq
->
data_len
,
cdrom_do_pc_continuation
);
return
startstop
;
}
...
...
@@ -1757,11 +1811,11 @@ ide_do_rw_cdrom (ide_drive_t *drive, struct request *rq, sector_t block)
if
(
blk_fs_request
(
rq
))
{
if
(
CDROM_CONFIG_FLAGS
(
drive
)
->
seeking
)
{
unsigned
long
el
pa
sed
=
jiffies
-
info
->
start_seek
;
unsigned
long
el
ap
sed
=
jiffies
-
info
->
start_seek
;
int
stat
=
HWIF
(
drive
)
->
INB
(
IDE_STATUS_REG
);
if
((
stat
&
SEEK_STAT
)
!=
SEEK_STAT
)
{
if
(
el
pa
sed
<
IDECD_SEEK_TIMEOUT
)
{
if
(
el
ap
sed
<
IDECD_SEEK_TIMEOUT
)
{
ide_stall_queue
(
drive
,
IDECD_SEEK_TIMER
);
return
ide_stopped
;
}
...
...
@@ -1781,14 +1835,14 @@ ide_do_rw_cdrom (ide_drive_t *drive, struct request *rq, sector_t block)
return
action
;
}
else
if
(
rq
->
flags
&
(
REQ_PC
|
REQ_SENSE
))
{
return
cdrom_do_packet_command
(
drive
);
}
else
if
(
rq
->
flags
&
REQ_BLOCK_PC
)
{
return
cdrom_do_block_pc
(
drive
,
rq
);
}
else
if
(
rq
->
flags
&
REQ_SPECIAL
)
{
/*
* right now this can only be a reset...
*/
cdrom_end_request
(
drive
,
1
);
return
ide_stopped
;
}
else
if
(
rq
->
flags
&
REQ_BLOCK_PC
)
{
return
cdrom_do_block_pc
(
drive
,
rq
);
}
blk_dump_rq_flags
(
rq
,
"ide-cd bad flags"
);
...
...
@@ -1853,23 +1907,23 @@ int msf_to_lba (byte m, byte s, byte f)
static
int
cdrom_check_status
(
ide_drive_t
*
drive
,
struct
request_sense
*
sense
)
{
struct
packet_command
pc
;
struct
request
req
;
struct
cdrom_info
*
info
=
drive
->
driver_data
;
struct
cdrom_device_info
*
cdi
=
&
info
->
devinfo
;
memset
(
&
pc
,
0
,
sizeof
(
pc
));
pc
.
sense
=
sense
;
cdrom_prepare_request
(
&
req
);
pc
.
c
[
0
]
=
GPCMD_TEST_UNIT_READY
;
req
.
sense
=
sense
;
req
.
cmd
[
0
]
=
GPCMD_TEST_UNIT_READY
;
#if ! STANDARD_ATAPI
/* the Sanyo 3 CD changer uses byte 7 of TEST_UNIT_READY to
switch CDs instead of supporting the LOAD_UNLOAD opcode */
pc
.
c
[
7
]
=
cdi
->
sanyo_slot
%
3
;
req
.
cmd
[
7
]
=
cdi
->
sanyo_slot
%
3
;
#endif
/* not STANDARD_ATAPI */
return
cdrom_queue_packet_command
(
drive
,
&
pc
);
return
cdrom_queue_packet_command
(
drive
,
&
req
);
}
...
...
@@ -1878,7 +1932,7 @@ static int
cdrom_lockdoor
(
ide_drive_t
*
drive
,
int
lockflag
,
struct
request_sense
*
sense
)
{
struct
request_sense
my_sense
;
struct
packet_command
pc
;
struct
request
req
;
int
stat
;
if
(
sense
==
NULL
)
...
...
@@ -1888,11 +1942,11 @@ cdrom_lockdoor(ide_drive_t *drive, int lockflag, struct request_sense *sense)
if
(
CDROM_CONFIG_FLAGS
(
drive
)
->
no_doorlock
)
{
stat
=
0
;
}
else
{
memset
(
&
pc
,
0
,
sizeof
(
pc
)
);
pc
.
sense
=
sense
;
pc
.
c
[
0
]
=
GPCMD_PREVENT_ALLOW_MEDIUM_REMOVAL
;
pc
.
c
[
4
]
=
lockflag
?
1
:
0
;
stat
=
cdrom_queue_packet_command
(
drive
,
&
pc
);
cdrom_prepare_request
(
&
req
);
req
.
sense
=
sense
;
req
.
cmd
[
0
]
=
GPCMD_PREVENT_ALLOW_MEDIUM_REMOVAL
;
req
.
cmd
[
4
]
=
lockflag
?
1
:
0
;
stat
=
cdrom_queue_packet_command
(
drive
,
&
req
);
}
/* If we got an illegal field error, the drive
...
...
@@ -1922,7 +1976,7 @@ cdrom_lockdoor(ide_drive_t *drive, int lockflag, struct request_sense *sense)
static
int
cdrom_eject
(
ide_drive_t
*
drive
,
int
ejectflag
,
struct
request_sense
*
sense
)
{
struct
packet_command
pc
;
struct
request
req
;
if
(
CDROM_CONFIG_FLAGS
(
drive
)
->
no_eject
&&
!
ejectflag
)
return
-
EDRIVE_CANT_DO_THIS
;
...
...
@@ -1931,12 +1985,12 @@ static int cdrom_eject(ide_drive_t *drive, int ejectflag,
if
(
CDROM_STATE_FLAGS
(
drive
)
->
door_locked
&&
ejectflag
)
return
0
;
memset
(
&
pc
,
0
,
sizeof
(
pc
));
pc
.
sense
=
sense
;
cdrom_prepare_request
(
&
req
);
pc
.
c
[
0
]
=
GPCMD_START_STOP_UNIT
;
pc
.
c
[
4
]
=
0x02
+
(
ejectflag
!=
0
);
return
cdrom_queue_packet_command
(
drive
,
&
pc
);
req
.
sense
=
sense
;
req
.
cmd
[
0
]
=
GPCMD_START_STOP_UNIT
;
req
.
cmd
[
4
]
=
0x02
+
(
ejectflag
!=
0
);
return
cdrom_queue_packet_command
(
drive
,
&
req
);
}
static
int
cdrom_read_capacity
(
ide_drive_t
*
drive
,
unsigned
long
*
capacity
,
...
...
@@ -1948,16 +2002,16 @@ static int cdrom_read_capacity(ide_drive_t *drive, unsigned long *capacity,
}
capbuf
;
int
stat
;
struct
packet_command
pc
;
struct
request
req
;
memset
(
&
pc
,
0
,
sizeof
(
pc
));
pc
.
sense
=
sense
;
cdrom_prepare_request
(
&
req
);
pc
.
c
[
0
]
=
GPCMD_READ_CDVD_CAPACITY
;
pc
.
buffer
=
(
char
*
)
&
capbuf
;
pc
.
buflen
=
sizeof
(
capbuf
);
req
.
sense
=
sense
;
req
.
cmd
[
0
]
=
GPCMD_READ_CDVD_CAPACITY
;
req
.
data
=
(
char
*
)
&
capbuf
;
req
.
data_len
=
sizeof
(
capbuf
);
stat
=
cdrom_queue_packet_command
(
drive
,
&
pc
);
stat
=
cdrom_queue_packet_command
(
drive
,
&
req
);
if
(
stat
==
0
)
*
capacity
=
1
+
be32_to_cpu
(
capbuf
.
lba
);
...
...
@@ -1968,24 +2022,24 @@ static int cdrom_read_tocentry(ide_drive_t *drive, int trackno, int msf_flag,
int
format
,
char
*
buf
,
int
buflen
,
struct
request_sense
*
sense
)
{
struct
packet_command
pc
;
struct
request
req
;
memset
(
&
pc
,
0
,
sizeof
(
pc
));
pc
.
sense
=
sense
;
cdrom_prepare_request
(
&
req
);
pc
.
buffer
=
buf
;
pc
.
buflen
=
buflen
;
pc
.
quiet
=
1
;
pc
.
c
[
0
]
=
GPCMD_READ_TOC_PMA_ATIP
;
pc
.
c
[
6
]
=
trackno
;
pc
.
c
[
7
]
=
(
buflen
>>
8
);
pc
.
c
[
8
]
=
(
buflen
&
0xff
);
pc
.
c
[
9
]
=
(
format
<<
6
);
req
.
sense
=
sense
;
req
.
data
=
buf
;
req
.
data_len
=
buflen
;
req
.
flags
|=
REQ_QUIET
;
req
.
cmd
[
0
]
=
GPCMD_READ_TOC_PMA_ATIP
;
req
.
cmd
[
6
]
=
trackno
;
req
.
cmd
[
7
]
=
(
buflen
>>
8
);
req
.
cmd
[
8
]
=
(
buflen
&
0xff
);
req
.
cmd
[
9
]
=
(
format
<<
6
);
if
(
msf_flag
)
pc
.
c
[
1
]
=
2
;
req
.
cmd
[
1
]
=
2
;
return
cdrom_queue_packet_command
(
drive
,
&
pc
);
return
cdrom_queue_packet_command
(
drive
,
&
req
);
}
...
...
@@ -2144,20 +2198,20 @@ static int cdrom_read_toc(ide_drive_t *drive, struct request_sense *sense)
static
int
cdrom_read_subchannel
(
ide_drive_t
*
drive
,
int
format
,
char
*
buf
,
int
buflen
,
struct
request_sense
*
sense
)
{
struct
packet_command
pc
;
struct
request
req
;
memset
(
&
pc
,
0
,
sizeof
(
pc
));
pc
.
sense
=
sense
;
cdrom_prepare_request
(
&
req
);
pc
.
buffer
=
buf
;
pc
.
buflen
=
buflen
;
pc
.
c
[
0
]
=
GPCMD_READ_SUBCHANNEL
;
pc
.
c
[
1
]
=
2
;
/* MSF addressing */
pc
.
c
[
2
]
=
0x40
;
/* request subQ data */
pc
.
c
[
3
]
=
format
;
pc
.
c
[
7
]
=
(
buflen
>>
8
);
pc
.
c
[
8
]
=
(
buflen
&
0xff
);
return
cdrom_queue_packet_command
(
drive
,
&
pc
);
req
.
sense
=
sense
;
req
.
data
=
buf
;
req
.
data_len
=
buflen
;
req
.
cmd
[
0
]
=
GPCMD_READ_SUBCHANNEL
;
req
.
cmd
[
1
]
=
2
;
/* MSF addressing */
req
.
cmd
[
2
]
=
0x40
;
/* request subQ data */
req
.
cmd
[
3
]
=
format
;
req
.
cmd
[
7
]
=
(
buflen
>>
8
);
req
.
cmd
[
8
]
=
(
buflen
&
0xff
);
return
cdrom_queue_packet_command
(
drive
,
&
req
);
}
/* ATAPI cdrom drives are free to select the speed you request or any slower
...
...
@@ -2165,45 +2219,45 @@ static int cdrom_read_subchannel(ide_drive_t *drive, int format, char *buf,
static
int
cdrom_select_speed
(
ide_drive_t
*
drive
,
int
speed
,
struct
request_sense
*
sense
)
{
struct
packet_command
pc
;
memset
(
&
pc
,
0
,
sizeof
(
pc
));
pc
.
sense
=
sense
;
struct
request
req
;
cdrom_prepare_request
(
&
req
);
req
.
sense
=
sense
;
if
(
speed
==
0
)
speed
=
0xffff
;
/* set to max */
else
speed
*=
177
;
/* Nx to kbytes/s */
pc
.
c
[
0
]
=
GPCMD_SET_SPEED
;
req
.
cmd
[
0
]
=
GPCMD_SET_SPEED
;
/* Read Drive speed in kbytes/second MSB */
pc
.
c
[
2
]
=
(
speed
>>
8
)
&
0xff
;
req
.
cmd
[
2
]
=
(
speed
>>
8
)
&
0xff
;
/* Read Drive speed in kbytes/second LSB */
pc
.
c
[
3
]
=
speed
&
0xff
;
req
.
cmd
[
3
]
=
speed
&
0xff
;
if
(
CDROM_CONFIG_FLAGS
(
drive
)
->
cd_r
||
CDROM_CONFIG_FLAGS
(
drive
)
->
cd_rw
||
CDROM_CONFIG_FLAGS
(
drive
)
->
dvd_r
)
{
/* Write Drive speed in kbytes/second MSB */
pc
.
c
[
4
]
=
(
speed
>>
8
)
&
0xff
;
req
.
cmd
[
4
]
=
(
speed
>>
8
)
&
0xff
;
/* Write Drive speed in kbytes/second LSB */
pc
.
c
[
5
]
=
speed
&
0xff
;
req
.
cmd
[
5
]
=
speed
&
0xff
;
}
return
cdrom_queue_packet_command
(
drive
,
&
pc
);
return
cdrom_queue_packet_command
(
drive
,
&
req
);
}
static
int
cdrom_play_audio
(
ide_drive_t
*
drive
,
int
lba_start
,
int
lba_end
)
{
struct
request_sense
sense
;
struct
packet_command
pc
;
struct
request
req
;
memset
(
&
pc
,
0
,
sizeof
(
pc
));
pc
.
sense
=
&
sense
;
cdrom_prepare_request
(
&
req
);
pc
.
c
[
0
]
=
GPCMD_PLAY_AUDIO_MSF
;
lba_to_msf
(
lba_start
,
&
pc
.
c
[
3
],
&
pc
.
c
[
4
],
&
pc
.
c
[
5
]);
lba_to_msf
(
lba_end
-
1
,
&
pc
.
c
[
6
],
&
pc
.
c
[
7
],
&
pc
.
c
[
8
]);
req
.
sense
=
&
sense
;
req
.
cmd
[
0
]
=
GPCMD_PLAY_AUDIO_MSF
;
lba_to_msf
(
lba_start
,
&
req
.
cmd
[
3
],
&
req
.
cmd
[
4
],
&
req
.
cmd
[
5
]);
lba_to_msf
(
lba_end
-
1
,
&
req
.
cmd
[
6
],
&
req
.
cmd
[
7
],
&
req
.
cmd
[
8
]);
return
cdrom_queue_packet_command
(
drive
,
&
pc
);
return
cdrom_queue_packet_command
(
drive
,
&
req
);
}
static
int
cdrom_get_toc_entry
(
ide_drive_t
*
drive
,
int
track
,
...
...
@@ -2237,7 +2291,7 @@ static int cdrom_get_toc_entry(ide_drive_t *drive, int track,
static
int
ide_cdrom_packet
(
struct
cdrom_device_info
*
cdi
,
struct
cdrom_generic_command
*
cgc
)
{
struct
packet_command
pc
;
struct
request
req
;
ide_drive_t
*
drive
=
(
ide_drive_t
*
)
cdi
->
handle
;
if
(
cgc
->
timeout
<=
0
)
...
...
@@ -2246,18 +2300,21 @@ static int ide_cdrom_packet(struct cdrom_device_info *cdi,
/* here we queue the commands from the uniform CD-ROM
layer. the packet must be complete, as we do not
touch it at all. */
memset
(
&
pc
,
0
,
sizeof
(
pc
)
);
memcpy
(
pc
.
c
,
cgc
->
cmd
,
CDROM_PACKET_SIZE
);
cdrom_prepare_request
(
&
req
);
memcpy
(
req
.
cmd
,
cgc
->
cmd
,
CDROM_PACKET_SIZE
);
if
(
cgc
->
sense
)
memset
(
cgc
->
sense
,
0
,
sizeof
(
struct
request_sense
));
pc
.
buffer
=
cgc
->
buffer
;
pc
.
buflen
=
cgc
->
buflen
;
pc
.
quiet
=
cgc
->
quiet
;
pc
.
timeout
=
cgc
->
timeout
;
pc
.
sense
=
cgc
->
sense
;
cgc
->
stat
=
cdrom_queue_packet_command
(
drive
,
&
pc
);
req
.
data
=
cgc
->
buffer
;
req
.
data_len
=
cgc
->
buflen
;
req
.
timeout
=
cgc
->
timeout
;
if
(
cgc
->
quiet
)
req
.
flags
|=
REQ_QUIET
;
req
.
sense
=
cgc
->
sense
;
cgc
->
stat
=
cdrom_queue_packet_command
(
drive
,
&
req
);
if
(
!
cgc
->
stat
)
cgc
->
buflen
-=
pc
.
buf
len
;
cgc
->
buflen
-=
req
.
data_
len
;
return
cgc
->
stat
;
}
...
...
@@ -2393,7 +2450,7 @@ int ide_cdrom_reset (struct cdrom_device_info *cdi)
struct
request
req
;
int
ret
;
ide_init_drive_cmd
(
&
req
);
cdrom_prepare_request
(
&
req
);
req
.
flags
=
REQ_SPECIAL
;
ret
=
ide_do_drive_cmd
(
drive
,
&
req
,
ide_wait
);
...
...
@@ -2969,6 +3026,14 @@ int ide_cdrom_ioctl (ide_drive_t *drive,
struct
inode
*
inode
,
struct
file
*
file
,
unsigned
int
cmd
,
unsigned
long
arg
)
{
int
error
;
/* Try the generic SCSI command ioctl's first.. */
error
=
scsi_cmd_ioctl
(
inode
->
i_bdev
,
cmd
,
arg
);
if
(
error
!=
-
ENOTTY
)
return
error
;
/* Then the generic cdrom ioctl's.. */
return
cdrom_ioctl
(
inode
,
file
,
cmd
,
arg
);
}
...
...
drivers/ide/ide-cd.h
View file @
40d1b71b
...
...
@@ -105,13 +105,6 @@ struct ide_cd_state_flags {
#define CDROM_STATE_FLAGS(drive) (&(((struct cdrom_info *)(drive->driver_data))->state_flags))
struct
packet_command
{
char
*
buffer
;
int
buflen
;
int
stat
;
int
quiet
;
int
timeout
;
struct
request_sense
*
sense
;
unsigned
char
c
[
12
];
};
/* Structure of a MSF cdrom address. */
...
...
drivers/ide/ide.c
View file @
40d1b71b
...
...
@@ -2639,7 +2639,7 @@ static int ide_ioctl (struct inode *inode, struct file *file,
case
CDROMEJECT
:
case
CDROMCLOSETRAY
:
return
block
_ioctl
(
inode
->
i_bdev
,
cmd
,
arg
);
return
scsi_cmd
_ioctl
(
inode
->
i_bdev
,
cmd
,
arg
);
case
HDIO_GET_BUSSTATE
:
if
(
!
capable
(
CAP_SYS_ADMIN
))
...
...
fs/block_dev.c
View file @
40d1b71b
...
...
@@ -824,44 +824,9 @@ static int blkdev_ioctl(struct inode *inode, struct file *file, unsigned cmd,
unsigned
long
arg
)
{
struct
block_device
*
bdev
=
inode
->
i_bdev
;
int
ret
=
-
EINVAL
;
switch
(
cmd
)
{
/*
* deprecated, use the /proc/iosched interface instead
*/
case
BLKELVGET
:
case
BLKELVSET
:
ret
=
-
ENOTTY
;
break
;
case
BLKRAGET
:
case
BLKROGET
:
case
BLKBSZGET
:
case
BLKSSZGET
:
case
BLKFRAGET
:
case
BLKSECTGET
:
case
BLKRASET
:
case
BLKFRASET
:
case
BLKBSZSET
:
case
BLKPG
:
ret
=
blk_ioctl
(
bdev
,
cmd
,
arg
);
break
;
case
BLKRRPART
:
ret
=
blkdev_reread_part
(
bdev
);
break
;
default:
if
(
bdev
->
bd_op
->
ioctl
)
ret
=
bdev
->
bd_op
->
ioctl
(
inode
,
file
,
cmd
,
arg
);
if
(
ret
==
-
EINVAL
)
{
switch
(
cmd
)
{
case
BLKGETSIZE
:
case
BLKGETSIZE64
:
case
BLKFLSBUF
:
case
BLKROSET
:
ret
=
blk_ioctl
(
bdev
,
cmd
,
arg
);
break
;
}
}
}
int
ret
=
blk_ioctl
(
bdev
,
cmd
,
arg
);
if
(
ret
==
-
ENOTTY
&&
bdev
->
bd_op
->
ioctl
)
ret
=
bdev
->
bd_op
->
ioctl
(
inode
,
file
,
cmd
,
arg
);
return
ret
;
}
...
...
include/linux/blkdev.h
View file @
40d1b71b
...
...
@@ -60,6 +60,12 @@ struct request {
int
tag
;
void
*
special
;
char
*
buffer
;
/* For packet commands */
unsigned
int
data_len
;
void
*
data
,
*
sense
;
unsigned
int
timeout
;
struct
completion
*
waiting
;
struct
bio
*
bio
,
*
biotail
;
request_queue_t
*
q
;
...
...
@@ -85,6 +91,8 @@ enum rq_flag_bits {
__REQ_BLOCK_PC
,
/* queued down pc from block layer */
__REQ_SENSE
,
/* sense retrival */
__REQ_FAILED
,
/* set if the request failed */
__REQ_QUIET
,
/* don't worry about errors */
__REQ_SPECIAL
,
/* driver suplied command */
__REQ_DRIVE_CMD
,
__REQ_DRIVE_TASK
,
...
...
@@ -103,6 +111,8 @@ enum rq_flag_bits {
#define REQ_PC (1 << __REQ_PC)
#define REQ_BLOCK_PC (1 << __REQ_BLOCK_PC)
#define REQ_SENSE (1 << __REQ_SENSE)
#define REQ_FAILED (1 << __REQ_FAILED)
#define REQ_QUIET (1 << __REQ_QUIET)
#define REQ_SPECIAL (1 << __REQ_SPECIAL)
#define REQ_DRIVE_CMD (1 << __REQ_DRIVE_CMD)
#define REQ_DRIVE_TASK (1 << __REQ_DRIVE_TASK)
...
...
@@ -301,7 +311,7 @@ extern int blk_remove_plug(request_queue_t *);
extern
void
blk_recount_segments
(
request_queue_t
*
,
struct
bio
*
);
extern
inline
int
blk_phys_contig_segment
(
request_queue_t
*
q
,
struct
bio
*
,
struct
bio
*
);
extern
inline
int
blk_hw_contig_segment
(
request_queue_t
*
q
,
struct
bio
*
,
struct
bio
*
);
extern
int
block
_ioctl
(
struct
block_device
*
,
unsigned
int
,
unsigned
long
);
extern
int
scsi_cmd
_ioctl
(
struct
block_device
*
,
unsigned
int
,
unsigned
long
);
extern
void
blk_start_queue
(
request_queue_t
*
q
);
extern
void
blk_stop_queue
(
request_queue_t
*
q
);
extern
void
__blk_stop_queue
(
request_queue_t
*
q
);
...
...
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