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
9bc97c63
Commit
9bc97c63
authored
Aug 03, 2004
by
David S. Miller
Browse files
Options
Browse Files
Download
Plain Diff
Merge nuts.davemloft.net:/disk1/BK/network-2.6
into nuts.davemloft.net:/disk1/BK/net-2.6
parents
d06a2c66
30e74fea
Changes
12
Hide whitespace changes
Inline
Side-by-side
Showing
12 changed files
with
249 additions
and
98 deletions
+249
-98
Makefile
Makefile
+1
-1
arch/ppc/kernel/head_e500.S
arch/ppc/kernel/head_e500.S
+2
-0
arch/ppc/kernel/irq.c
arch/ppc/kernel/irq.c
+6
-1
arch/ppc/platforms/pmac_pic.c
arch/ppc/platforms/pmac_pic.c
+28
-16
drivers/block/ll_rw_blk.c
drivers/block/ll_rw_blk.c
+33
-43
drivers/block/scsi_ioctl.c
drivers/block/scsi_ioctl.c
+1
-1
drivers/cdrom/cdrom.c
drivers/cdrom/cdrom.c
+1
-1
drivers/ide/ide-cd.c
drivers/ide/ide-cd.c
+11
-7
drivers/macintosh/therm_pm72.c
drivers/macintosh/therm_pm72.c
+21
-0
fs/bio.c
fs/bio.c
+140
-26
include/linux/bio.h
include/linux/bio.h
+4
-1
include/linux/blkdev.h
include/linux/blkdev.h
+1
-1
No files found.
Makefile
View file @
9bc97c63
VERSION
=
2
VERSION
=
2
PATCHLEVEL
=
6
PATCHLEVEL
=
6
SUBLEVEL
=
8
SUBLEVEL
=
8
EXTRAVERSION
=
-rc
2
EXTRAVERSION
=
-rc
3
NAME
=
Zonked Quokka
NAME
=
Zonked Quokka
# *DOCUMENTATION*
# *DOCUMENTATION*
...
...
arch/ppc/kernel/head_e500.S
View file @
9bc97c63
...
@@ -1177,6 +1177,8 @@ _GLOBAL(giveup_spe)
...
@@ -1177,6 +1177,8 @@ _GLOBAL(giveup_spe)
evmwumiaa
evr6
,
evr6
,
evr6
/*
evr6
<-
ACC
=
0
*
0
+
ACC
*/
evmwumiaa
evr6
,
evr6
,
evr6
/*
evr6
<-
ACC
=
0
*
0
+
ACC
*/
li
r4
,
THREAD_ACC
li
r4
,
THREAD_ACC
evstddx
evr6
,
r4
,
r3
/*
save
off
accumulator
*/
evstddx
evr6
,
r4
,
r3
/*
save
off
accumulator
*/
mfspr
r6
,
SPRN_SPEFSCR
stw
r6
,
THREAD_SPEFSCR
(
r3
)
/*
save
spefscr
register
value
*/
beq
1
f
beq
1
f
lwz
r4
,
_MSR
-
STACK_FRAME_OVERHEAD
(
r5
)
lwz
r4
,
_MSR
-
STACK_FRAME_OVERHEAD
(
r5
)
lis
r3
,
MSR_SPE
@
h
lis
r3
,
MSR_SPE
@
h
...
...
arch/ppc/kernel/irq.c
View file @
9bc97c63
...
@@ -171,7 +171,12 @@ setup_irq(unsigned int irq, struct irqaction * new)
...
@@ -171,7 +171,12 @@ setup_irq(unsigned int irq, struct irqaction * new)
if
(
!
shared
)
{
if
(
!
shared
)
{
desc
->
depth
=
0
;
desc
->
depth
=
0
;
desc
->
status
&=
~
(
IRQ_DISABLED
|
IRQ_AUTODETECT
|
IRQ_WAITING
);
desc
->
status
&=
~
(
IRQ_DISABLED
|
IRQ_AUTODETECT
|
IRQ_WAITING
);
unmask_irq
(
irq
);
if
(
desc
->
handler
)
{
if
(
desc
->
handler
->
startup
)
desc
->
handler
->
startup
(
irq
);
else
if
(
desc
->
handler
->
enable
)
desc
->
handler
->
enable
(
irq
);
}
}
}
spin_unlock_irqrestore
(
&
desc
->
lock
,
flags
);
spin_unlock_irqrestore
(
&
desc
->
lock
,
flags
);
...
...
arch/ppc/platforms/pmac_pic.c
View file @
9bc97c63
...
@@ -144,6 +144,22 @@ static void __pmac pmac_set_irq_mask(unsigned int irq_nr, int nokicklost)
...
@@ -144,6 +144,22 @@ static void __pmac pmac_set_irq_mask(unsigned int irq_nr, int nokicklost)
spin_unlock_irqrestore
(
&
pmac_pic_lock
,
flags
);
spin_unlock_irqrestore
(
&
pmac_pic_lock
,
flags
);
}
}
/* When an irq gets requested for the first client, if it's an
* edge interrupt, we clear any previous one on the controller
*/
static
unsigned
int
__pmac
pmac_startup_irq
(
unsigned
int
irq_nr
)
{
unsigned
long
bit
=
1UL
<<
(
irq_nr
&
0x1f
);
int
i
=
irq_nr
>>
5
;
if
((
irq_desc
[
irq_nr
].
status
&
IRQ_LEVEL
)
==
0
)
out_le32
(
&
pmac_irq_hw
[
i
]
->
ack
,
bit
);
set_bit
(
irq_nr
,
ppc_cached_irq_mask
);
pmac_set_irq_mask
(
irq_nr
,
0
);
return
0
;
}
static
void
__pmac
pmac_mask_irq
(
unsigned
int
irq_nr
)
static
void
__pmac
pmac_mask_irq
(
unsigned
int
irq_nr
)
{
{
clear_bit
(
irq_nr
,
ppc_cached_irq_mask
);
clear_bit
(
irq_nr
,
ppc_cached_irq_mask
);
...
@@ -168,25 +184,21 @@ static void __pmac pmac_end_irq(unsigned int irq_nr)
...
@@ -168,25 +184,21 @@ static void __pmac pmac_end_irq(unsigned int irq_nr)
struct
hw_interrupt_type
pmac_pic
=
{
struct
hw_interrupt_type
pmac_pic
=
{
" PMAC-PIC "
,
.
typename
=
" PMAC-PIC "
,
NULL
,
.
startup
=
pmac_startup_irq
,
NULL
,
.
enable
=
pmac_unmask_irq
,
pmac_unmask_irq
,
.
disable
=
pmac_mask_irq
,
pmac_mask_irq
,
.
ack
=
pmac_mask_and_ack_irq
,
pmac_mask_and_ack_irq
,
.
end
=
pmac_end_irq
,
pmac_end_irq
,
NULL
};
};
struct
hw_interrupt_type
gatwick_pic
=
{
struct
hw_interrupt_type
gatwick_pic
=
{
" GATWICK "
,
.
typename
=
" GATWICK "
,
NULL
,
.
startup
=
pmac_startup_irq
,
NULL
,
.
enable
=
pmac_unmask_irq
,
pmac_unmask_irq
,
.
disable
=
pmac_mask_irq
,
pmac_mask_irq
,
.
ack
=
pmac_mask_and_ack_irq
,
pmac_mask_and_ack_irq
,
.
end
=
pmac_end_irq
,
pmac_end_irq
,
NULL
};
};
static
irqreturn_t
gatwick_action
(
int
cpl
,
void
*
dev_id
,
struct
pt_regs
*
regs
)
static
irqreturn_t
gatwick_action
(
int
cpl
,
void
*
dev_id
,
struct
pt_regs
*
regs
)
...
...
drivers/block/ll_rw_blk.c
View file @
9bc97c63
...
@@ -1823,50 +1823,43 @@ EXPORT_SYMBOL(blk_insert_request);
...
@@ -1823,50 +1823,43 @@ EXPORT_SYMBOL(blk_insert_request);
struct
request
*
blk_rq_map_user
(
request_queue_t
*
q
,
int
rw
,
void
__user
*
ubuf
,
struct
request
*
blk_rq_map_user
(
request_queue_t
*
q
,
int
rw
,
void
__user
*
ubuf
,
unsigned
int
len
)
unsigned
int
len
)
{
{
struct
request
*
rq
=
NULL
;
unsigned
long
uaddr
;
char
*
buf
=
NULL
;
struct
request
*
rq
;
struct
bio
*
bio
;
struct
bio
*
bio
;
int
ret
;
if
(
len
>
(
q
->
max_sectors
<<
9
))
return
ERR_PTR
(
-
EINVAL
);
if
((
!
len
&&
ubuf
)
||
(
len
&&
!
ubuf
))
return
ERR_PTR
(
-
EINVAL
);
rq
=
blk_get_request
(
q
,
rw
,
__GFP_WAIT
);
rq
=
blk_get_request
(
q
,
rw
,
__GFP_WAIT
);
if
(
!
rq
)
if
(
!
rq
)
return
ERR_PTR
(
-
ENOMEM
);
return
ERR_PTR
(
-
ENOMEM
);
bio
=
bio_map_user
(
q
,
NULL
,
(
unsigned
long
)
ubuf
,
len
,
rw
==
READ
);
/*
if
(
!
bio
)
{
* if alignment requirement is satisfied, map in user pages for
int
bytes
=
(
len
+
511
)
&
~
511
;
* direct dma. else, set up kernel bounce buffers
*/
buf
=
kmalloc
(
bytes
,
q
->
bounce_gfp
|
GFP_USER
);
uaddr
=
(
unsigned
long
)
ubuf
;
if
(
!
buf
)
{
if
(
!
(
uaddr
&
queue_dma_alignment
(
q
))
&&
!
(
len
&
queue_dma_alignment
(
q
)))
ret
=
-
ENOMEM
;
bio
=
bio_map_user
(
q
,
NULL
,
uaddr
,
len
,
rw
==
READ
);
goto
fault
;
else
}
bio
=
bio_copy_user
(
q
,
uaddr
,
len
,
rw
==
READ
);
if
(
rw
==
WRITE
)
{
if
(
copy_from_user
(
buf
,
ubuf
,
len
))
{
ret
=
-
EFAULT
;
goto
fault
;
}
}
else
memset
(
buf
,
0
,
len
);
}
rq
->
bio
=
rq
->
biotail
=
bio
;
if
(
!
IS_ERR
(
bio
))
{
if
(
rq
->
bio
)
rq
->
bio
=
rq
->
biotail
=
bio
;
blk_rq_bio_prep
(
q
,
rq
,
bio
);
blk_rq_bio_prep
(
q
,
rq
,
bio
);
rq
->
buffer
=
rq
->
data
=
buf
;
rq
->
buffer
=
rq
->
data
=
NULL
;
rq
->
data_len
=
len
;
rq
->
data_len
=
len
;
return
rq
;
return
rq
;
fault:
}
if
(
buf
)
kfree
(
buf
);
if
(
bio
)
bio_unmap_user
(
bio
,
1
);
if
(
rq
)
blk_put_request
(
rq
);
return
ERR_PTR
(
ret
);
/*
* bio is the err-ptr
*/
blk_put_request
(
rq
);
return
(
struct
request
*
)
bio
;
}
}
EXPORT_SYMBOL
(
blk_rq_map_user
);
EXPORT_SYMBOL
(
blk_rq_map_user
);
...
@@ -1880,18 +1873,15 @@ EXPORT_SYMBOL(blk_rq_map_user);
...
@@ -1880,18 +1873,15 @@ EXPORT_SYMBOL(blk_rq_map_user);
* Description:
* Description:
* Unmap a request previously mapped by blk_rq_map_user().
* Unmap a request previously mapped by blk_rq_map_user().
*/
*/
int
blk_rq_unmap_user
(
struct
request
*
rq
,
void
__user
*
ubuf
,
struct
bio
*
bio
,
int
blk_rq_unmap_user
(
struct
request
*
rq
,
struct
bio
*
bio
,
unsigned
int
ulen
)
unsigned
int
ulen
)
{
{
const
int
read
=
rq_data_dir
(
rq
)
==
READ
;
int
ret
=
0
;
int
ret
=
0
;
if
(
bio
)
if
(
bio
)
{
bio_unmap_user
(
bio
,
read
);
if
(
bio_flagged
(
bio
,
BIO_USER_MAPPED
))
if
(
rq
->
buffer
)
{
bio_unmap_user
(
bio
);
if
(
read
&&
copy_to_user
(
ubuf
,
rq
->
buffer
,
ulen
))
else
ret
=
-
EFAULT
;
ret
=
bio_uncopy_user
(
bio
);
kfree
(
rq
->
buffer
);
}
}
blk_put_request
(
rq
);
blk_put_request
(
rq
);
...
...
drivers/block/scsi_ioctl.c
View file @
9bc97c63
...
@@ -211,7 +211,7 @@ static int sg_io(request_queue_t *q, struct gendisk *bd_disk,
...
@@ -211,7 +211,7 @@ static int sg_io(request_queue_t *q, struct gendisk *bd_disk,
hdr
->
sb_len_wr
=
len
;
hdr
->
sb_len_wr
=
len
;
}
}
if
(
blk_rq_unmap_user
(
rq
,
hdr
->
dxferp
,
bio
,
hdr
->
dxfer_len
))
if
(
blk_rq_unmap_user
(
rq
,
bio
,
hdr
->
dxfer_len
))
return
-
EFAULT
;
return
-
EFAULT
;
/* may not have succeeded, but output values written to control
/* may not have succeeded, but output values written to control
...
...
drivers/cdrom/cdrom.c
View file @
9bc97c63
...
@@ -2015,7 +2015,7 @@ static int cdrom_read_cdda_bpc(struct cdrom_device_info *cdi, __u8 __user *ubuf,
...
@@ -2015,7 +2015,7 @@ static int cdrom_read_cdda_bpc(struct cdrom_device_info *cdi, __u8 __user *ubuf,
cdi
->
last_sense
=
s
->
sense_key
;
cdi
->
last_sense
=
s
->
sense_key
;
}
}
if
(
blk_rq_unmap_user
(
rq
,
ubuf
,
bio
,
len
))
if
(
blk_rq_unmap_user
(
rq
,
bio
,
len
))
ret
=
-
EFAULT
;
ret
=
-
EFAULT
;
if
(
ret
)
if
(
ret
)
...
...
drivers/ide/ide-cd.c
View file @
9bc97c63
...
@@ -1959,13 +1959,17 @@ static ide_startstop_t cdrom_do_block_pc(ide_drive_t *drive, struct request *rq)
...
@@ -1959,13 +1959,17 @@ static ide_startstop_t cdrom_do_block_pc(ide_drive_t *drive, struct request *rq)
* sg request
* sg request
*/
*/
if
(
rq
->
bio
)
{
if
(
rq
->
bio
)
{
if
(
rq
->
data_len
&
3
)
{
int
mask
=
drive
->
queue
->
dma_alignment
;
printk
(
"%s: block pc not aligned, len=%d
\n
"
,
drive
->
name
,
rq
->
data_len
);
unsigned
long
addr
=
(
unsigned
long
)
page_address
(
bio_page
(
rq
->
bio
));
cdrom_end_request
(
drive
,
0
);
return
ide_stopped
;
}
info
->
dma
=
drive
->
using_dma
;
info
->
cmd
=
rq_data_dir
(
rq
);
info
->
cmd
=
rq_data_dir
(
rq
);
info
->
dma
=
drive
->
using_dma
;
/*
* check if dma is safe
*/
if
((
rq
->
data_len
&
mask
)
||
(
addr
&
mask
))
info
->
dma
=
0
;
}
}
/* Start sending the command to the drive. */
/* Start sending the command to the drive. */
...
@@ -3133,7 +3137,7 @@ int ide_cdrom_setup (ide_drive_t *drive)
...
@@ -3133,7 +3137,7 @@ int ide_cdrom_setup (ide_drive_t *drive)
int
nslots
;
int
nslots
;
blk_queue_prep_rq
(
drive
->
queue
,
ide_cdrom_prep_fn
);
blk_queue_prep_rq
(
drive
->
queue
,
ide_cdrom_prep_fn
);
blk_queue_dma_alignment
(
drive
->
queue
,
3
);
blk_queue_dma_alignment
(
drive
->
queue
,
3
1
);
drive
->
queue
->
unplug_delay
=
(
1
*
HZ
)
/
1000
;
drive
->
queue
->
unplug_delay
=
(
1
*
HZ
)
/
1000
;
if
(
!
drive
->
queue
->
unplug_delay
)
if
(
!
drive
->
queue
->
unplug_delay
)
drive
->
queue
->
unplug_delay
=
1
;
drive
->
queue
->
unplug_delay
=
1
;
...
...
drivers/macintosh/therm_pm72.c
View file @
9bc97c63
...
@@ -317,6 +317,20 @@ static int fan_write_reg(int reg, const unsigned char *ptr, int nb)
...
@@ -317,6 +317,20 @@ static int fan_write_reg(int reg, const unsigned char *ptr, int nb)
return
nw
;
return
nw
;
}
}
static
int
start_fcu
(
void
)
{
unsigned
char
buf
=
0xff
;
int
rc
;
rc
=
fan_write_reg
(
0xe
,
&
buf
,
1
);
if
(
rc
<
0
)
return
-
EIO
;
rc
=
fan_write_reg
(
0x2e
,
&
buf
,
1
);
if
(
rc
<
0
)
return
-
EIO
;
return
0
;
}
static
int
set_rpm_fan
(
int
fan
,
int
rpm
)
static
int
set_rpm_fan
(
int
fan
,
int
rpm
)
{
{
unsigned
char
buf
[
2
];
unsigned
char
buf
[
2
];
...
@@ -1011,6 +1025,12 @@ static int main_control_loop(void *x)
...
@@ -1011,6 +1025,12 @@ static int main_control_loop(void *x)
down
(
&
driver_lock
);
down
(
&
driver_lock
);
if
(
start_fcu
()
<
0
)
{
printk
(
KERN_ERR
"kfand: failed to start FCU
\n
"
);
up
(
&
driver_lock
);
goto
out
;
}
/* Set the PCI fan once for now */
/* Set the PCI fan once for now */
set_pwm_fan
(
SLOTS_FAN_PWM_ID
,
SLOTS_FAN_DEFAULT_PWM
);
set_pwm_fan
(
SLOTS_FAN_PWM_ID
,
SLOTS_FAN_DEFAULT_PWM
);
...
@@ -1057,6 +1077,7 @@ static int main_control_loop(void *x)
...
@@ -1057,6 +1077,7 @@ static int main_control_loop(void *x)
schedule_timeout
(
HZ
-
elapsed
);
schedule_timeout
(
HZ
-
elapsed
);
}
}
out:
DBG
(
"main_control_loop ended
\n
"
);
DBG
(
"main_control_loop ended
\n
"
);
ctrl_task
=
0
;
ctrl_task
=
0
;
...
...
fs/bio.c
View file @
9bc97c63
...
@@ -376,6 +376,115 @@ int bio_add_page(struct bio *bio, struct page *page, unsigned int len,
...
@@ -376,6 +376,115 @@ int bio_add_page(struct bio *bio, struct page *page, unsigned int len,
len
,
offset
);
len
,
offset
);
}
}
/**
* bio_uncopy_user - finish previously mapped bio
* @bio: bio being terminated
*
* Free pages allocated from bio_copy_user() and write back data
* to user space in case of a read.
*/
int
bio_uncopy_user
(
struct
bio
*
bio
)
{
struct
bio_vec
*
bvec
;
int
i
,
ret
=
0
;
if
(
bio_data_dir
(
bio
)
==
READ
)
{
char
*
uaddr
=
bio
->
bi_private
;
__bio_for_each_segment
(
bvec
,
bio
,
i
,
0
)
{
char
*
addr
=
page_address
(
bvec
->
bv_page
);
if
(
!
ret
&&
copy_to_user
(
uaddr
,
addr
,
bvec
->
bv_len
))
ret
=
-
EFAULT
;
__free_page
(
bvec
->
bv_page
);
uaddr
+=
bvec
->
bv_len
;
}
}
bio_put
(
bio
);
return
ret
;
}
/**
* bio_copy_user - copy user data to bio
* @q: destination block queue
* @uaddr: start of user address
* @len: length in bytes
* @write_to_vm: bool indicating writing to pages or not
*
* Prepares and returns a bio for indirect user io, bouncing data
* to/from kernel pages as necessary. Must be paired with
* call bio_uncopy_user() on io completion.
*/
struct
bio
*
bio_copy_user
(
request_queue_t
*
q
,
unsigned
long
uaddr
,
unsigned
int
len
,
int
write_to_vm
)
{
unsigned
long
end
=
(
uaddr
+
len
+
PAGE_SIZE
-
1
)
>>
PAGE_SHIFT
;
unsigned
long
start
=
uaddr
>>
PAGE_SHIFT
;
struct
bio_vec
*
bvec
;
struct
page
*
page
;
struct
bio
*
bio
;
int
i
,
ret
;
bio
=
bio_alloc
(
GFP_KERNEL
,
end
-
start
);
if
(
!
bio
)
return
ERR_PTR
(
-
ENOMEM
);
ret
=
0
;
while
(
len
)
{
unsigned
int
bytes
=
PAGE_SIZE
;
if
(
bytes
>
len
)
bytes
=
len
;
page
=
alloc_page
(
q
->
bounce_gfp
|
GFP_KERNEL
);
if
(
!
page
)
{
ret
=
-
ENOMEM
;
break
;
}
if
(
__bio_add_page
(
q
,
bio
,
page
,
bytes
,
0
)
<
bytes
)
{
ret
=
-
EINVAL
;
break
;
}
len
-=
bytes
;
}
/*
* success
*/
if
(
!
ret
)
{
if
(
!
write_to_vm
)
{
bio
->
bi_rw
|=
(
1
<<
BIO_RW
);
/*
* for a write, copy in data to kernel pages
*/
ret
=
-
EFAULT
;
bio_for_each_segment
(
bvec
,
bio
,
i
)
{
char
*
addr
=
page_address
(
bvec
->
bv_page
);
if
(
copy_from_user
(
addr
,
(
char
*
)
uaddr
,
bvec
->
bv_len
))
goto
cleanup
;
}
}
bio
->
bi_private
=
(
void
*
)
uaddr
;
return
bio
;
}
/*
* cleanup
*/
cleanup:
bio_for_each_segment
(
bvec
,
bio
,
i
)
__free_page
(
bvec
->
bv_page
);
bio_put
(
bio
);
return
ERR_PTR
(
ret
);
}
static
struct
bio
*
__bio_map_user
(
request_queue_t
*
q
,
struct
block_device
*
bdev
,
static
struct
bio
*
__bio_map_user
(
request_queue_t
*
q
,
struct
block_device
*
bdev
,
unsigned
long
uaddr
,
unsigned
int
len
,
unsigned
long
uaddr
,
unsigned
int
len
,
int
write_to_vm
)
int
write_to_vm
)
...
@@ -392,12 +501,13 @@ static struct bio *__bio_map_user(request_queue_t *q, struct block_device *bdev,
...
@@ -392,12 +501,13 @@ static struct bio *__bio_map_user(request_queue_t *q, struct block_device *bdev,
* size for now, in the future we can relax this restriction
* size for now, in the future we can relax this restriction
*/
*/
if
((
uaddr
&
queue_dma_alignment
(
q
))
||
(
len
&
queue_dma_alignment
(
q
)))
if
((
uaddr
&
queue_dma_alignment
(
q
))
||
(
len
&
queue_dma_alignment
(
q
)))
return
NULL
;
return
ERR_PTR
(
-
EINVAL
)
;
bio
=
bio_alloc
(
GFP_KERNEL
,
nr_pages
);
bio
=
bio_alloc
(
GFP_KERNEL
,
nr_pages
);
if
(
!
bio
)
if
(
!
bio
)
return
NULL
;
return
ERR_PTR
(
-
ENOMEM
)
;
ret
=
-
ENOMEM
;
pages
=
kmalloc
(
nr_pages
*
sizeof
(
struct
page
*
),
GFP_KERNEL
);
pages
=
kmalloc
(
nr_pages
*
sizeof
(
struct
page
*
),
GFP_KERNEL
);
if
(
!
pages
)
if
(
!
pages
)
goto
out
;
goto
out
;
...
@@ -446,11 +556,12 @@ static struct bio *__bio_map_user(request_queue_t *q, struct block_device *bdev,
...
@@ -446,11 +556,12 @@ static struct bio *__bio_map_user(request_queue_t *q, struct block_device *bdev,
if
(
!
write_to_vm
)
if
(
!
write_to_vm
)
bio
->
bi_rw
|=
(
1
<<
BIO_RW
);
bio
->
bi_rw
|=
(
1
<<
BIO_RW
);
bio
->
bi_flags
|=
(
1
<<
BIO_USER_MAPPED
);
return
bio
;
return
bio
;
out:
out:
kfree
(
pages
);
kfree
(
pages
);
bio_put
(
bio
);
bio_put
(
bio
);
return
NULL
;
return
ERR_PTR
(
ret
)
;
}
}
/**
/**
...
@@ -461,7 +572,7 @@ static struct bio *__bio_map_user(request_queue_t *q, struct block_device *bdev,
...
@@ -461,7 +572,7 @@ static struct bio *__bio_map_user(request_queue_t *q, struct block_device *bdev,
* @write_to_vm: bool indicating writing to pages or not
* @write_to_vm: bool indicating writing to pages or not
*
*
* Map the user space address into a bio suitable for io to a block
* Map the user space address into a bio suitable for io to a block
* device.
* device.
Returns an error pointer in case of error.
*/
*/
struct
bio
*
bio_map_user
(
request_queue_t
*
q
,
struct
block_device
*
bdev
,
struct
bio
*
bio_map_user
(
request_queue_t
*
q
,
struct
block_device
*
bdev
,
unsigned
long
uaddr
,
unsigned
int
len
,
int
write_to_vm
)
unsigned
long
uaddr
,
unsigned
int
len
,
int
write_to_vm
)
...
@@ -470,26 +581,29 @@ struct bio *bio_map_user(request_queue_t *q, struct block_device *bdev,
...
@@ -470,26 +581,29 @@ struct bio *bio_map_user(request_queue_t *q, struct block_device *bdev,
bio
=
__bio_map_user
(
q
,
bdev
,
uaddr
,
len
,
write_to_vm
);
bio
=
__bio_map_user
(
q
,
bdev
,
uaddr
,
len
,
write_to_vm
);
if
(
bio
)
{
if
(
IS_ERR
(
bio
))
/*
return
bio
;
* subtle -- if __bio_map_user() ended up bouncing a bio,
* it would normally disappear when its bi_end_io is run.
* however, we need it for the unmap, so grab an extra
* reference to it
*/
bio_get
(
bio
);
if
(
bio
->
bi_size
<
len
)
{
/*
bio_endio
(
bio
,
bio
->
bi_size
,
0
);
* subtle -- if __bio_map_user() ended up bouncing a bio,
bio_unmap_user
(
bio
,
0
);
* it would normally disappear when its bi_end_io is run.
return
NULL
;
* however, we need it for the unmap, so grab an extra
}
* reference to it
}
*/
bio_get
(
bio
);
return
bio
;
if
(
bio
->
bi_size
==
len
)
return
bio
;
/*
* don't support partial mappings
*/
bio_endio
(
bio
,
bio
->
bi_size
,
0
);
bio_unmap_user
(
bio
);
return
ERR_PTR
(
-
EINVAL
);
}
}
static
void
__bio_unmap_user
(
struct
bio
*
bio
,
int
write_to_vm
)
static
void
__bio_unmap_user
(
struct
bio
*
bio
)
{
{
struct
bio_vec
*
bvec
;
struct
bio_vec
*
bvec
;
int
i
;
int
i
;
...
@@ -510,7 +624,7 @@ static void __bio_unmap_user(struct bio *bio, int write_to_vm)
...
@@ -510,7 +624,7 @@ static void __bio_unmap_user(struct bio *bio, int write_to_vm)
* make sure we dirty pages we wrote to
* make sure we dirty pages we wrote to
*/
*/
__bio_for_each_segment
(
bvec
,
bio
,
i
,
0
)
{
__bio_for_each_segment
(
bvec
,
bio
,
i
,
0
)
{
if
(
write_to_vm
)
if
(
bio_data_dir
(
bio
)
==
READ
)
set_page_dirty_lock
(
bvec
->
bv_page
);
set_page_dirty_lock
(
bvec
->
bv_page
);
page_cache_release
(
bvec
->
bv_page
);
page_cache_release
(
bvec
->
bv_page
);
...
@@ -522,17 +636,15 @@ static void __bio_unmap_user(struct bio *bio, int write_to_vm)
...
@@ -522,17 +636,15 @@ static void __bio_unmap_user(struct bio *bio, int write_to_vm)
/**
/**
* bio_unmap_user - unmap a bio
* bio_unmap_user - unmap a bio
* @bio: the bio being unmapped
* @bio: the bio being unmapped
* @write_to_vm: bool indicating whether pages were written to
*
*
* Unmap a bio previously mapped by bio_map_user(). The @write_to_vm
* Unmap a bio previously mapped by bio_map_user(). Must be called with
* must be the same as passed into bio_map_user(). Must be called with
* a process context.
* a process context.
*
*
* bio_unmap_user() may sleep.
* bio_unmap_user() may sleep.
*/
*/
void
bio_unmap_user
(
struct
bio
*
bio
,
int
write_to_vm
)
void
bio_unmap_user
(
struct
bio
*
bio
)
{
{
__bio_unmap_user
(
bio
,
write_to_vm
);
__bio_unmap_user
(
bio
);
bio_put
(
bio
);
bio_put
(
bio
);
}
}
...
@@ -863,3 +975,5 @@ EXPORT_SYMBOL(bio_unmap_user);
...
@@ -863,3 +975,5 @@ EXPORT_SYMBOL(bio_unmap_user);
EXPORT_SYMBOL
(
bio_pair_release
);
EXPORT_SYMBOL
(
bio_pair_release
);
EXPORT_SYMBOL
(
bio_split
);
EXPORT_SYMBOL
(
bio_split
);
EXPORT_SYMBOL
(
bio_split_pool
);
EXPORT_SYMBOL
(
bio_split_pool
);
EXPORT_SYMBOL
(
bio_copy_user
);
EXPORT_SYMBOL
(
bio_uncopy_user
);
include/linux/bio.h
View file @
9bc97c63
...
@@ -120,6 +120,7 @@ struct bio {
...
@@ -120,6 +120,7 @@ struct bio {
#define BIO_SEG_VALID 3
/* nr_hw_seg valid */
#define BIO_SEG_VALID 3
/* nr_hw_seg valid */
#define BIO_CLONED 4
/* doesn't own data */
#define BIO_CLONED 4
/* doesn't own data */
#define BIO_BOUNCED 5
/* bio is a bounce bio */
#define BIO_BOUNCED 5
/* bio is a bounce bio */
#define BIO_USER_MAPPED 6
/* contains user pages */
#define bio_flagged(bio, flag) ((bio)->bi_flags & (1 << (flag)))
#define bio_flagged(bio, flag) ((bio)->bi_flags & (1 << (flag)))
/*
/*
...
@@ -264,9 +265,11 @@ extern int bio_add_page(struct bio *, struct page *, unsigned int,unsigned int);
...
@@ -264,9 +265,11 @@ extern int bio_add_page(struct bio *, struct page *, unsigned int,unsigned int);
extern
int
bio_get_nr_vecs
(
struct
block_device
*
);
extern
int
bio_get_nr_vecs
(
struct
block_device
*
);
extern
struct
bio
*
bio_map_user
(
struct
request_queue
*
,
struct
block_device
*
,
extern
struct
bio
*
bio_map_user
(
struct
request_queue
*
,
struct
block_device
*
,
unsigned
long
,
unsigned
int
,
int
);
unsigned
long
,
unsigned
int
,
int
);
extern
void
bio_unmap_user
(
struct
bio
*
,
int
);
extern
void
bio_unmap_user
(
struct
bio
*
);
extern
void
bio_set_pages_dirty
(
struct
bio
*
bio
);
extern
void
bio_set_pages_dirty
(
struct
bio
*
bio
);
extern
void
bio_check_pages_dirty
(
struct
bio
*
bio
);
extern
void
bio_check_pages_dirty
(
struct
bio
*
bio
);
extern
struct
bio
*
bio_copy_user
(
struct
request_queue
*
,
unsigned
long
,
unsigned
int
,
int
);
extern
int
bio_uncopy_user
(
struct
bio
*
);
#ifdef CONFIG_HIGHMEM
#ifdef CONFIG_HIGHMEM
/*
/*
...
...
include/linux/blkdev.h
View file @
9bc97c63
...
@@ -524,7 +524,7 @@ extern void __blk_stop_queue(request_queue_t *q);
...
@@ -524,7 +524,7 @@ extern void __blk_stop_queue(request_queue_t *q);
extern
void
blk_run_queue
(
request_queue_t
*
);
extern
void
blk_run_queue
(
request_queue_t
*
);
extern
void
blk_queue_activity_fn
(
request_queue_t
*
,
activity_fn
*
,
void
*
);
extern
void
blk_queue_activity_fn
(
request_queue_t
*
,
activity_fn
*
,
void
*
);
extern
struct
request
*
blk_rq_map_user
(
request_queue_t
*
,
int
,
void
__user
*
,
unsigned
int
);
extern
struct
request
*
blk_rq_map_user
(
request_queue_t
*
,
int
,
void
__user
*
,
unsigned
int
);
extern
int
blk_rq_unmap_user
(
struct
request
*
,
void
__user
*
,
struct
bio
*
,
unsigned
int
);
extern
int
blk_rq_unmap_user
(
struct
request
*
,
struct
bio
*
,
unsigned
int
);
extern
int
blk_execute_rq
(
request_queue_t
*
,
struct
gendisk
*
,
struct
request
*
);
extern
int
blk_execute_rq
(
request_queue_t
*
,
struct
gendisk
*
,
struct
request
*
);
static
inline
request_queue_t
*
bdev_get_queue
(
struct
block_device
*
bdev
)
static
inline
request_queue_t
*
bdev_get_queue
(
struct
block_device
*
bdev
)
...
...
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