Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
88e1f06c
Commit
88e1f06c
authored
Sep 07, 2004
by
Linus Torvalds
Browse files
Options
Browse Files
Download
Plain Diff
Merge
bk://bk.arm.linux.org.uk/linux-2.6-mmc
into ppc970.osdl.org:/home/torvalds/v2.6/linux
parents
32f6e431
b0e7d727
Changes
8
Show whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
236 additions
and
88 deletions
+236
-88
drivers/mmc/mmc.c
drivers/mmc/mmc.c
+17
-9
drivers/mmc/mmc_block.c
drivers/mmc/mmc_block.c
+9
-14
drivers/mmc/mmc_queue.c
drivers/mmc/mmc_queue.c
+7
-2
drivers/mmc/mmci.c
drivers/mmc/mmci.c
+175
-51
drivers/mmc/mmci.h
drivers/mmc/mmci.h
+6
-11
drivers/mmc/pxamci.c
drivers/mmc/pxamci.c
+5
-0
include/linux/mmc/host.h
include/linux/mmc/host.h
+7
-1
include/linux/mmc/mmc.h
include/linux/mmc/mmc.h
+10
-0
No files found.
drivers/mmc/mmc.c
View file @
88e1f06c
/*
/*
* linux/drivers/mmc/mmc.c
* linux/drivers/mmc/mmc.c
*
*
* Copyright (C) 2003 Russell King, All Rights Reserved.
* Copyright (C) 2003
-2004
Russell King, All Rights Reserved.
*
*
* This program is free software; you can redistribute it and/or modify
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* it under the terms of the GNU General Public License version 2 as
...
@@ -14,6 +14,7 @@
...
@@ -14,6 +14,7 @@
#include <linux/completion.h>
#include <linux/completion.h>
#include <linux/device.h>
#include <linux/device.h>
#include <linux/delay.h>
#include <linux/delay.h>
#include <linux/pagemap.h>
#include <linux/err.h>
#include <linux/err.h>
#include <linux/mmc/card.h>
#include <linux/mmc/card.h>
...
@@ -212,7 +213,7 @@ int __mmc_claim_host(struct mmc_host *host, struct mmc_card *card)
...
@@ -212,7 +213,7 @@ int __mmc_claim_host(struct mmc_host *host, struct mmc_card *card)
cmd
.
opcode
=
MMC_SELECT_CARD
;
cmd
.
opcode
=
MMC_SELECT_CARD
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
flags
=
MMC_RSP_
SHORT
|
MMC_RSP_CRC
;
cmd
.
flags
=
MMC_RSP_
R1
;
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
}
}
...
@@ -430,7 +431,7 @@ static int mmc_send_op_cond(struct mmc_host *host, u32 ocr, u32 *rocr)
...
@@ -430,7 +431,7 @@ static int mmc_send_op_cond(struct mmc_host *host, u32 ocr, u32 *rocr)
cmd
.
opcode
=
MMC_SEND_OP_COND
;
cmd
.
opcode
=
MMC_SEND_OP_COND
;
cmd
.
arg
=
ocr
;
cmd
.
arg
=
ocr
;
cmd
.
flags
=
MMC_RSP_
SHORT
;
cmd
.
flags
=
MMC_RSP_
R3
;
for
(
i
=
100
;
i
;
i
--
)
{
for
(
i
=
100
;
i
;
i
--
)
{
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
0
);
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
0
);
...
@@ -468,7 +469,7 @@ static void mmc_discover_cards(struct mmc_host *host)
...
@@ -468,7 +469,7 @@ static void mmc_discover_cards(struct mmc_host *host)
cmd
.
opcode
=
MMC_ALL_SEND_CID
;
cmd
.
opcode
=
MMC_ALL_SEND_CID
;
cmd
.
arg
=
0
;
cmd
.
arg
=
0
;
cmd
.
flags
=
MMC_RSP_
LONG
|
MMC_RSP_CRC
;
cmd
.
flags
=
MMC_RSP_
R2
;
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
if
(
err
==
MMC_ERR_TIMEOUT
)
{
if
(
err
==
MMC_ERR_TIMEOUT
)
{
...
@@ -497,7 +498,7 @@ static void mmc_discover_cards(struct mmc_host *host)
...
@@ -497,7 +498,7 @@ static void mmc_discover_cards(struct mmc_host *host)
cmd
.
opcode
=
MMC_SET_RELATIVE_ADDR
;
cmd
.
opcode
=
MMC_SET_RELATIVE_ADDR
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
flags
=
MMC_RSP_
SHORT
|
MMC_RSP_CRC
;
cmd
.
flags
=
MMC_RSP_
R1
;
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
if
(
err
!=
MMC_ERR_NONE
)
if
(
err
!=
MMC_ERR_NONE
)
...
@@ -518,7 +519,7 @@ static void mmc_read_csds(struct mmc_host *host)
...
@@ -518,7 +519,7 @@ static void mmc_read_csds(struct mmc_host *host)
cmd
.
opcode
=
MMC_SEND_CSD
;
cmd
.
opcode
=
MMC_SEND_CSD
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
flags
=
MMC_RSP_
LONG
|
MMC_RSP_CRC
;
cmd
.
flags
=
MMC_RSP_
R2
;
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
if
(
err
!=
MMC_ERR_NONE
)
{
if
(
err
!=
MMC_ERR_NONE
)
{
...
@@ -566,7 +567,7 @@ static void mmc_check_cards(struct mmc_host *host)
...
@@ -566,7 +567,7 @@ static void mmc_check_cards(struct mmc_host *host)
cmd
.
opcode
=
MMC_SEND_STATUS
;
cmd
.
opcode
=
MMC_SEND_STATUS
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
flags
=
MMC_RSP_
SHORT
|
MMC_RSP_CRC
;
cmd
.
flags
=
MMC_RSP_
R1
;
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
err
=
mmc_wait_for_cmd
(
host
,
&
cmd
,
CMD_RETRIES
);
if
(
err
==
MMC_ERR_NONE
)
if
(
err
==
MMC_ERR_NONE
)
...
@@ -715,14 +716,21 @@ struct mmc_host *mmc_alloc_host(int extra, struct device *dev)
...
@@ -715,14 +716,21 @@ struct mmc_host *mmc_alloc_host(int extra, struct device *dev)
if
(
host
)
{
if
(
host
)
{
memset
(
host
,
0
,
sizeof
(
struct
mmc_host
)
+
extra
);
memset
(
host
,
0
,
sizeof
(
struct
mmc_host
)
+
extra
);
host
->
priv
=
host
+
1
;
spin_lock_init
(
&
host
->
lock
);
spin_lock_init
(
&
host
->
lock
);
init_waitqueue_head
(
&
host
->
wq
);
init_waitqueue_head
(
&
host
->
wq
);
INIT_LIST_HEAD
(
&
host
->
cards
);
INIT_LIST_HEAD
(
&
host
->
cards
);
INIT_WORK
(
&
host
->
detect
,
mmc_rescan
,
host
);
INIT_WORK
(
&
host
->
detect
,
mmc_rescan
,
host
);
host
->
dev
=
dev
;
host
->
dev
=
dev
;
/*
* By default, hosts do not support SGIO or large requests.
* They have to set these according to their abilities.
*/
host
->
max_hw_segs
=
1
;
host
->
max_phys_segs
=
1
;
host
->
max_sectors
=
1
<<
(
PAGE_CACHE_SHIFT
-
9
);
host
->
max_seg_size
=
PAGE_CACHE_SIZE
;
}
}
return
host
;
return
host
;
...
...
drivers/mmc/mmc_block.c
View file @
88e1f06c
...
@@ -43,7 +43,6 @@
...
@@ -43,7 +43,6 @@
#define MMC_SHIFT 3
#define MMC_SHIFT 3
static
int
mmc_major
;
static
int
mmc_major
;
static
int
maxsectors
=
8
;
/*
/*
* There is one mmc_blk_data per slot.
* There is one mmc_blk_data per slot.
...
@@ -180,22 +179,22 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
...
@@ -180,22 +179,22 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
brq
.
mrq
.
data
=
&
brq
.
data
;
brq
.
mrq
.
data
=
&
brq
.
data
;
brq
.
cmd
.
arg
=
req
->
sector
<<
9
;
brq
.
cmd
.
arg
=
req
->
sector
<<
9
;
brq
.
cmd
.
flags
=
MMC_RSP_
SHORT
|
MMC_RSP_CRC
;
brq
.
cmd
.
flags
=
MMC_RSP_
R1
;
brq
.
data
.
req
=
req
;
brq
.
data
.
req
=
req
;
brq
.
data
.
timeout_ns
=
card
->
csd
.
tacc_ns
*
10
;
brq
.
data
.
timeout_ns
=
card
->
csd
.
tacc_ns
*
10
;
brq
.
data
.
timeout_clks
=
card
->
csd
.
tacc_clks
*
10
;
brq
.
data
.
timeout_clks
=
card
->
csd
.
tacc_clks
*
10
;
brq
.
data
.
blksz_bits
=
md
->
block_bits
;
brq
.
data
.
blksz_bits
=
md
->
block_bits
;
brq
.
data
.
blocks
=
req
->
current_
nr_sectors
>>
(
md
->
block_bits
-
9
);
brq
.
data
.
blocks
=
req
->
nr_sectors
>>
(
md
->
block_bits
-
9
);
brq
.
stop
.
opcode
=
MMC_STOP_TRANSMISSION
;
brq
.
stop
.
opcode
=
MMC_STOP_TRANSMISSION
;
brq
.
stop
.
arg
=
0
;
brq
.
stop
.
arg
=
0
;
brq
.
stop
.
flags
=
MMC_RSP_
SHORT
|
MMC_RSP_CRC
|
MMC_RSP_BUSY
;
brq
.
stop
.
flags
=
MMC_RSP_
R1B
;
if
(
rq_data_dir
(
req
)
==
READ
)
{
if
(
rq_data_dir
(
req
)
==
READ
)
{
brq
.
cmd
.
opcode
=
brq
.
data
.
blocks
>
1
?
MMC_READ_MULTIPLE_BLOCK
:
MMC_READ_SINGLE_BLOCK
;
brq
.
cmd
.
opcode
=
brq
.
data
.
blocks
>
1
?
MMC_READ_MULTIPLE_BLOCK
:
MMC_READ_SINGLE_BLOCK
;
brq
.
data
.
flags
|=
MMC_DATA_READ
;
brq
.
data
.
flags
|=
MMC_DATA_READ
;
}
else
{
}
else
{
brq
.
cmd
.
opcode
=
MMC_WRITE_BLOCK
;
brq
.
cmd
.
opcode
=
MMC_WRITE_BLOCK
;
brq
.
cmd
.
flags
|=
MMC_RSP_BUSY
;
brq
.
cmd
.
flags
=
MMC_RSP_R1B
;
brq
.
data
.
flags
|=
MMC_DATA_WRITE
;
brq
.
data
.
flags
|=
MMC_DATA_WRITE
;
brq
.
data
.
blocks
=
1
;
brq
.
data
.
blocks
=
1
;
}
}
...
@@ -225,7 +224,7 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
...
@@ -225,7 +224,7 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
cmd
.
opcode
=
MMC_SEND_STATUS
;
cmd
.
opcode
=
MMC_SEND_STATUS
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
arg
=
card
->
rca
<<
16
;
cmd
.
flags
=
MMC_RSP_
SHORT
|
MMC_RSP_CRC
;
cmd
.
flags
=
MMC_RSP_
R1
;
err
=
mmc_wait_for_cmd
(
card
->
host
,
&
cmd
,
5
);
err
=
mmc_wait_for_cmd
(
card
->
host
,
&
cmd
,
5
);
if
(
err
)
{
if
(
err
)
{
printk
(
KERN_ERR
"%s: error %d requesting status
\n
"
,
printk
(
KERN_ERR
"%s: error %d requesting status
\n
"
,
...
@@ -334,11 +333,10 @@ static struct mmc_blk_data *mmc_blk_alloc(struct mmc_card *card)
...
@@ -334,11 +333,10 @@ static struct mmc_blk_data *mmc_blk_alloc(struct mmc_card *card)
sprintf
(
md
->
disk
->
disk_name
,
"mmcblk%d"
,
devidx
);
sprintf
(
md
->
disk
->
disk_name
,
"mmcblk%d"
,
devidx
);
sprintf
(
md
->
disk
->
devfs_name
,
"mmc/blk%d"
,
devidx
);
sprintf
(
md
->
disk
->
devfs_name
,
"mmc/blk%d"
,
devidx
);
md
->
block_bits
=
md
->
queue
.
card
->
csd
.
read_blkbits
;
md
->
block_bits
=
card
->
csd
.
read_blkbits
;
blk_queue_max_sectors
(
md
->
queue
.
queue
,
maxsectors
);
blk_queue_hardsect_size
(
md
->
queue
.
queue
,
1
<<
md
->
block_bits
);
blk_queue_hardsect_size
(
md
->
queue
.
queue
,
1
<<
md
->
block_bits
);
set_capacity
(
md
->
disk
,
md
->
queue
.
card
->
csd
.
capacity
);
set_capacity
(
md
->
disk
,
card
->
csd
.
capacity
);
}
}
out:
out:
return
md
;
return
md
;
...
@@ -353,7 +351,7 @@ mmc_blk_set_blksize(struct mmc_blk_data *md, struct mmc_card *card)
...
@@ -353,7 +351,7 @@ mmc_blk_set_blksize(struct mmc_blk_data *md, struct mmc_card *card)
mmc_card_claim_host
(
card
);
mmc_card_claim_host
(
card
);
cmd
.
opcode
=
MMC_SET_BLOCKLEN
;
cmd
.
opcode
=
MMC_SET_BLOCKLEN
;
cmd
.
arg
=
1
<<
card
->
csd
.
read_blkbits
;
cmd
.
arg
=
1
<<
card
->
csd
.
read_blkbits
;
cmd
.
flags
=
MMC_RSP_
SHORT
|
MMC_RSP_CRC
;
cmd
.
flags
=
MMC_RSP_
R1
;
err
=
mmc_wait_for_cmd
(
card
->
host
,
&
cmd
,
5
);
err
=
mmc_wait_for_cmd
(
card
->
host
,
&
cmd
,
5
);
mmc_card_release_host
(
card
);
mmc_card_release_host
(
card
);
...
@@ -440,7 +438,7 @@ static int mmc_blk_resume(struct mmc_card *card)
...
@@ -440,7 +438,7 @@ static int mmc_blk_resume(struct mmc_card *card)
struct
mmc_blk_data
*
md
=
mmc_get_drvdata
(
card
);
struct
mmc_blk_data
*
md
=
mmc_get_drvdata
(
card
);
if
(
md
)
{
if
(
md
)
{
mmc_blk_set_blksize
(
md
,
md
->
queue
.
card
);
mmc_blk_set_blksize
(
md
,
card
);
blk_start_queue
(
md
->
queue
.
queue
);
blk_start_queue
(
md
->
queue
.
queue
);
}
}
return
0
;
return
0
;
...
@@ -489,9 +487,6 @@ static void __exit mmc_blk_exit(void)
...
@@ -489,9 +487,6 @@ static void __exit mmc_blk_exit(void)
module_init
(
mmc_blk_init
);
module_init
(
mmc_blk_init
);
module_exit
(
mmc_blk_exit
);
module_exit
(
mmc_blk_exit
);
module_param
(
maxsectors
,
int
,
0444
);
MODULE_PARM_DESC
(
maxsectors
,
"Maximum number of sectors for a single request"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_DESCRIPTION
(
"Multimedia Card (MMC) block device driver"
);
MODULE_DESCRIPTION
(
"Multimedia Card (MMC) block device driver"
);
drivers/mmc/mmc_queue.c
View file @
88e1f06c
...
@@ -124,11 +124,12 @@ static void mmc_request(request_queue_t *q)
...
@@ -124,11 +124,12 @@ static void mmc_request(request_queue_t *q)
*/
*/
int
mmc_init_queue
(
struct
mmc_queue
*
mq
,
struct
mmc_card
*
card
,
spinlock_t
*
lock
)
int
mmc_init_queue
(
struct
mmc_queue
*
mq
,
struct
mmc_card
*
card
,
spinlock_t
*
lock
)
{
{
struct
mmc_host
*
host
=
card
->
host
;
u64
limit
=
BLK_BOUNCE_HIGH
;
u64
limit
=
BLK_BOUNCE_HIGH
;
int
ret
;
int
ret
;
if
(
card
->
host
->
dev
->
dma_mask
&&
*
card
->
host
->
dev
->
dma_mask
)
if
(
host
->
dev
->
dma_mask
&&
*
host
->
dev
->
dma_mask
)
limit
=
*
card
->
host
->
dev
->
dma_mask
;
limit
=
*
host
->
dev
->
dma_mask
;
mq
->
card
=
card
;
mq
->
card
=
card
;
mq
->
queue
=
blk_init_queue
(
mmc_request
,
lock
);
mq
->
queue
=
blk_init_queue
(
mmc_request
,
lock
);
...
@@ -137,6 +138,10 @@ int mmc_init_queue(struct mmc_queue *mq, struct mmc_card *card, spinlock_t *lock
...
@@ -137,6 +138,10 @@ int mmc_init_queue(struct mmc_queue *mq, struct mmc_card *card, spinlock_t *lock
blk_queue_prep_rq
(
mq
->
queue
,
mmc_prep_request
);
blk_queue_prep_rq
(
mq
->
queue
,
mmc_prep_request
);
blk_queue_bounce_limit
(
mq
->
queue
,
limit
);
blk_queue_bounce_limit
(
mq
->
queue
,
limit
);
blk_queue_max_sectors
(
mq
->
queue
,
host
->
max_sectors
);
blk_queue_max_phys_segments
(
mq
->
queue
,
host
->
max_phys_segs
);
blk_queue_max_hw_segments
(
mq
->
queue
,
host
->
max_hw_segs
);
blk_queue_max_segment_size
(
mq
->
queue
,
host
->
max_seg_size
);
mq
->
queue
->
queuedata
=
mq
;
mq
->
queue
->
queuedata
=
mq
;
mq
->
req
=
NULL
;
mq
->
req
=
NULL
;
...
...
drivers/mmc/mmci.c
View file @
88e1f06c
...
@@ -43,10 +43,9 @@ static void
...
@@ -43,10 +43,9 @@ static void
mmci_request_end
(
struct
mmci_host
*
host
,
struct
mmc_request
*
mrq
)
mmci_request_end
(
struct
mmci_host
*
host
,
struct
mmc_request
*
mrq
)
{
{
writel
(
0
,
host
->
base
+
MMCICOMMAND
);
writel
(
0
,
host
->
base
+
MMCICOMMAND
);
host
->
mrq
=
NULL
;
host
->
mrq
=
NULL
;
host
->
cmd
=
NULL
;
host
->
cmd
=
NULL
;
host
->
data
=
NULL
;
host
->
buffer
=
NULL
;
if
(
mrq
->
data
)
if
(
mrq
->
data
)
mrq
->
data
->
bytes_xfered
=
host
->
data_xfered
;
mrq
->
data
->
bytes_xfered
=
host
->
data_xfered
;
...
@@ -60,6 +59,13 @@ mmci_request_end(struct mmci_host *host, struct mmc_request *mrq)
...
@@ -60,6 +59,13 @@ mmci_request_end(struct mmci_host *host, struct mmc_request *mrq)
spin_lock
(
&
host
->
lock
);
spin_lock
(
&
host
->
lock
);
}
}
static
void
mmci_stop_data
(
struct
mmci_host
*
host
)
{
writel
(
0
,
host
->
base
+
MMCIDATACTRL
);
writel
(
0
,
host
->
base
+
MMCIMASK1
);
host
->
data
=
NULL
;
}
static
void
mmci_start_data
(
struct
mmci_host
*
host
,
struct
mmc_data
*
data
)
static
void
mmci_start_data
(
struct
mmci_host
*
host
,
struct
mmc_data
*
data
)
{
{
unsigned
int
datactrl
,
timeout
,
irqmask
;
unsigned
int
datactrl
,
timeout
,
irqmask
;
...
@@ -69,7 +75,7 @@ static void mmci_start_data(struct mmci_host *host, struct mmc_data *data)
...
@@ -69,7 +75,7 @@ static void mmci_start_data(struct mmci_host *host, struct mmc_data *data)
1
<<
data
->
blksz_bits
,
data
->
blocks
,
data
->
flags
);
1
<<
data
->
blksz_bits
,
data
->
blocks
,
data
->
flags
);
host
->
data
=
data
;
host
->
data
=
data
;
host
->
buffer
=
data
->
req
->
buffer
;
host
->
offset
=
0
;
host
->
size
=
data
->
blocks
<<
data
->
blksz_bits
;
host
->
size
=
data
->
blocks
<<
data
->
blksz_bits
;
host
->
data_xfered
=
0
;
host
->
data_xfered
=
0
;
...
@@ -94,6 +100,7 @@ static void mmci_start_data(struct mmci_host *host, struct mmc_data *data)
...
@@ -94,6 +100,7 @@ static void mmci_start_data(struct mmci_host *host, struct mmc_data *data)
}
}
writel
(
datactrl
,
base
+
MMCIDATACTRL
);
writel
(
datactrl
,
base
+
MMCIDATACTRL
);
writel
(
readl
(
base
+
MMCIMASK0
)
&
~
MCI_DATAENDMASK
,
base
+
MMCIMASK0
);
writel
(
irqmask
,
base
+
MMCIMASK1
);
writel
(
irqmask
,
base
+
MMCIMASK1
);
}
}
...
@@ -147,7 +154,8 @@ mmci_data_irq(struct mmci_host *host, struct mmc_data *data,
...
@@ -147,7 +154,8 @@ mmci_data_irq(struct mmci_host *host, struct mmc_data *data,
status
|=
MCI_DATAEND
;
status
|=
MCI_DATAEND
;
}
}
if
(
status
&
MCI_DATAEND
)
{
if
(
status
&
MCI_DATAEND
)
{
host
->
data
=
NULL
;
mmci_stop_data
(
host
);
if
(
!
data
->
stop
)
{
if
(
!
data
->
stop
)
{
mmci_request_end
(
host
,
data
->
mrq
);
mmci_request_end
(
host
,
data
->
mrq
);
}
else
{
}
else
{
...
@@ -182,73 +190,172 @@ mmci_cmd_irq(struct mmci_host *host, struct mmc_command *cmd,
...
@@ -182,73 +190,172 @@ mmci_cmd_irq(struct mmci_host *host, struct mmc_command *cmd,
}
}
}
}
/*
static
int
mmci_pio_read
(
struct
mmci_host
*
host
,
struct
request
*
req
,
u32
status
)
* PIO data transfer IRQ handler.
*/
static
irqreturn_t
mmci_pio_irq
(
int
irq
,
void
*
dev_id
,
struct
pt_regs
*
regs
)
{
{
struct
mmci_host
*
host
=
dev_id
;
void
*
base
=
host
->
base
;
void
*
base
=
host
->
base
;
u32
status
;
int
ret
=
0
;
int
ret
=
0
;
do
{
do
{
status
=
readl
(
base
+
MMCISTATUS
);
unsigned
long
flags
;
unsigned
int
bio_remain
;
char
*
buffer
;
if
(
!
(
status
&
(
MCI_RXDATAAVLBL
|
MCI_RXFIFOHALFFULL
|
/*
MCI_TXFIFOHALFEMPTY
)))
* Check for data available.
*/
if
(
!
(
status
&
MCI_RXDATAAVLBL
))
break
;
break
;
DBG
(
host
,
"irq1 %08x
\n
"
,
status
);
/*
* Map the BIO buffer.
*/
buffer
=
bio_kmap_irq
(
req
->
cbio
,
&
flags
);
bio_remain
=
(
req
->
current_nr_sectors
<<
9
)
-
host
->
offset
;
do
{
int
count
=
host
->
size
-
(
readl
(
base
+
MMCIFIFOCNT
)
<<
2
);
if
(
status
&
(
MCI_RXDATAAVLBL
|
MCI_RXFIFOHALFFULL
))
{
if
(
count
>
bio_remain
)
unsigned
int
count
=
host
->
size
-
(
readl
(
base
+
MMCIFIFOCNT
)
<<
2
)
;
count
=
bio_remain
;
if
(
count
<
0
)
count
=
0
;
if
(
count
>
0
)
{
if
(
count
&&
host
->
buffer
)
{
ret
=
1
;
readsl
(
base
+
MMCIFIFO
,
host
->
buffer
,
count
>>
2
);
readsl
(
base
+
MMCIFIFO
,
buffer
+
host
->
offset
,
count
>>
2
);
host
->
buffer
+=
count
;
host
->
offset
+=
count
;
host
->
size
-=
count
;
host
->
size
-=
count
;
if
(
host
->
size
==
0
)
bio_remain
-=
count
;
host
->
buffer
=
NULL
;
if
(
bio_remain
==
0
)
}
else
{
goto
next_bio
;
static
int
first
=
1
;
if
(
first
)
{
first
=
0
;
printk
(
KERN_ERR
"MMCI: sinking excessive data
\n
"
);
}
readl
(
base
+
MMCIFIFO
);
}
}
}
status
=
readl
(
base
+
MMCISTATUS
);
}
while
(
status
&
MCI_RXDATAAVLBL
);
bio_kunmap_irq
(
buffer
,
&
flags
);
break
;
next_bio:
bio_kunmap_irq
(
buffer
,
&
flags
);
/*
* Ok, we've completed that BIO, move on to next
* BIO in the chain. Note: this doesn't actually
* complete the BIO!
*/
if
(
!
process_that_request_first
(
req
,
req
->
current_nr_sectors
))
break
;
host
->
offset
=
0
;
status
=
readl
(
base
+
MMCISTATUS
);
}
while
(
1
);
/*
* If we're nearing the end of the read, switch to
* "any data available" mode.
*/
if
(
host
->
size
<
MCI_FIFOSIZE
)
writel
(
MCI_RXDATAAVLBLMASK
,
base
+
MMCIMASK1
);
return
ret
;
}
static
int
mmci_pio_write
(
struct
mmci_host
*
host
,
struct
request
*
req
,
u32
status
)
{
void
*
base
=
host
->
base
;
int
ret
=
0
;
do
{
unsigned
long
flags
;
unsigned
int
bio_remain
;
char
*
buffer
;
/*
/*
* We only need to test the half-empty flag here - if
* We only need to test the half-empty flag here - if
* the FIFO is completely empty, then by definition
* the FIFO is completely empty, then by definition
* it is more than half empty.
* it is more than half empty.
*/
*/
if
(
status
&
MCI_TXFIFOHALFEMPTY
)
{
if
(
!
(
status
&
MCI_TXFIFOHALFEMPTY
))
unsigned
int
maxcnt
=
status
&
MCI_TXFIFOEMPTY
?
break
;
MCI_FIFOSIZE
:
MCI_FIFOHALFSIZE
;
unsigned
int
count
=
min
(
host
->
size
,
maxcnt
);
/*
* Map the BIO buffer.
*/
buffer
=
bio_kmap_irq
(
req
->
cbio
,
&
flags
);
bio_remain
=
(
req
->
current_nr_sectors
<<
9
)
-
host
->
offset
;
writesl
(
base
+
MMCIFIFO
,
host
->
buffer
,
count
>>
2
);
do
{
unsigned
int
count
,
maxcnt
;
maxcnt
=
status
&
MCI_TXFIFOEMPTY
?
MCI_FIFOSIZE
:
MCI_FIFOHALFSIZE
;
count
=
min
(
bio_remain
,
maxcnt
);
host
->
buffer
+=
count
;
writesl
(
base
+
MMCIFIFO
,
buffer
+
host
->
offset
,
count
>>
2
);
host
->
offset
+=
count
;
host
->
size
-=
count
;
host
->
size
-=
count
;
bio_remain
-=
count
;
ret
=
1
;
if
(
bio_remain
==
0
)
goto
next_bio
;
status
=
readl
(
base
+
MMCISTATUS
);
}
while
(
status
&
MCI_TXFIFOHALFEMPTY
);
bio_kunmap_irq
(
buffer
,
&
flags
);
break
;
next_bio:
bio_kunmap_irq
(
buffer
,
&
flags
);
/*
/*
* If we run out of data, disable the data IRQs;
* Ok, we've completed that BIO, move on to next
* this prevents a race where the FIFO becomes
* BIO in the chain. Note: this doesn't actually
* empty before the chip itself has disabled the
* complete the BIO!
* data path.
*/
*/
if
(
host
->
size
==
0
)
if
(
!
process_that_request_first
(
req
,
req
->
current_nr_sectors
))
break
;
host
->
offset
=
0
;
status
=
readl
(
base
+
MMCISTATUS
);
}
while
(
1
);
return
ret
;
}
/*
* PIO data transfer IRQ handler.
*/
static
irqreturn_t
mmci_pio_irq
(
int
irq
,
void
*
dev_id
,
struct
pt_regs
*
regs
)
{
struct
mmci_host
*
host
=
dev_id
;
struct
request
*
req
;
void
*
base
=
host
->
base
;
u32
status
;
int
ret
=
0
;
status
=
readl
(
base
+
MMCISTATUS
);
DBG
(
host
,
"irq1 %08x
\n
"
,
status
);
req
=
host
->
data
->
req
;
if
(
status
&
MCI_RXACTIVE
)
ret
=
mmci_pio_read
(
host
,
req
,
status
);
else
if
(
status
&
MCI_TXACTIVE
)
ret
=
mmci_pio_write
(
host
,
req
,
status
);
/*
* If we run out of data, disable the data IRQs; this
* prevents a race where the FIFO becomes empty before
* the chip itself has disabled the data path, and
* stops us racing with our data end IRQ.
*/
if
(
host
->
size
==
0
)
{
writel
(
0
,
base
+
MMCIMASK1
);
writel
(
0
,
base
+
MMCIMASK1
);
writel
(
readl
(
base
+
MMCIMASK0
)
|
MCI_DATAENDMASK
,
base
+
MMCIMASK0
);
}
}
ret
=
1
;
}
while
(
status
);
return
IRQ_RETVAL
(
ret
);
return
IRQ_RETVAL
(
ret
);
}
}
...
@@ -268,11 +375,9 @@ static irqreturn_t mmci_irq(int irq, void *dev_id, struct pt_regs *regs)
...
@@ -268,11 +375,9 @@ static irqreturn_t mmci_irq(int irq, void *dev_id, struct pt_regs *regs)
struct
mmc_data
*
data
;
struct
mmc_data
*
data
;
status
=
readl
(
host
->
base
+
MMCISTATUS
);
status
=
readl
(
host
->
base
+
MMCISTATUS
);
status
&=
readl
(
host
->
base
+
MMCIMASK0
);
writel
(
status
,
host
->
base
+
MMCICLEAR
);
writel
(
status
,
host
->
base
+
MMCICLEAR
);
if
(
!
(
status
&
MCI_IRQMASK
))
break
;
DBG
(
host
,
"irq0 %08x
\n
"
,
status
);
DBG
(
host
,
"irq0 %08x
\n
"
,
status
);
data
=
host
->
data
;
data
=
host
->
data
;
...
@@ -427,6 +532,25 @@ static int mmci_probe(struct amba_device *dev, void *id)
...
@@ -427,6 +532,25 @@ static int mmci_probe(struct amba_device *dev, void *id)
mmc
->
f_max
=
min
(
host
->
mclk
,
fmax
);
mmc
->
f_max
=
min
(
host
->
mclk
,
fmax
);
mmc
->
ocr_avail
=
plat
->
ocr_mask
;
mmc
->
ocr_avail
=
plat
->
ocr_mask
;
/*
* We can do SGIO
*/
mmc
->
max_hw_segs
=
16
;
mmc
->
max_phys_segs
=
16
;
/*
* Since we only have a 16-bit data length register, we must
* ensure that we don't exceed 2^16-1 bytes in a single request.
* Choose 64 (512-byte) sectors as the limit.
*/
mmc
->
max_sectors
=
64
;
/*
* Set the maximum segment size. Since we aren't doing DMA
* (yet) we are only limited by the data length register.
*/
mmc
->
max_seg_size
=
mmc
->
max_sectors
<<
9
;
spin_lock_init
(
&
host
->
lock
);
spin_lock_init
(
&
host
->
lock
);
writel
(
0
,
host
->
base
+
MMCIMASK0
);
writel
(
0
,
host
->
base
+
MMCIMASK0
);
...
...
drivers/mmc/mmci.h
View file @
88e1f06c
...
@@ -103,18 +103,15 @@
...
@@ -103,18 +103,15 @@
#define MMCIFIFOCNT 0x048
#define MMCIFIFOCNT 0x048
#define MMCIFIFO 0x080
/* to 0x0bc */
#define MMCIFIFO 0x080
/* to 0x0bc */
#define MCI_IRQMASK \
(MCI_CMDCRCFAIL|MCI_DATACRCFAIL|MCI_CMDTIMEOUT|MCI_DATATIMEOUT| \
MCI_TXUNDERRUN|MCI_RXOVERRUN|MCI_CMDRESPEND|MCI_CMDSENT| \
MCI_DATAEND|MCI_DATABLOCKEND)
#define MCI_IRQENABLE \
#define MCI_IRQENABLE \
(MCI_CMDCRCFAILMASK|MCI_DATACRCFAILMASK|MCI_CMDTIMEOUTMASK| \
(MCI_CMDCRCFAILMASK|MCI_DATACRCFAILMASK|MCI_CMDTIMEOUTMASK| \
MCI_DATATIMEOUTMASK|MCI_TXUNDERRUNMASK|MCI_RXOVERRUNMASK| \
MCI_DATATIMEOUTMASK|MCI_TXUNDERRUNMASK|MCI_RXOVERRUNMASK| \
MCI_CMDRESPENDMASK|MCI_CMDSENTMASK|MCI_DATAENDMASK| \
MCI_CMDRESPENDMASK|MCI_CMDSENTMASK|MCI_DATABLOCKENDMASK)
MCI_DATABLOCKENDMASK)
#define MCI_FIFOSIZE 16
/*
* The size of the FIFO in bytes.
*/
#define MCI_FIFOSIZE (16*4)
#define MCI_FIFOHALFSIZE (MCI_FIFOSIZE / 2)
#define MCI_FIFOHALFSIZE (MCI_FIFOSIZE / 2)
...
@@ -141,8 +138,6 @@ struct mmci_host {
...
@@ -141,8 +138,6 @@ struct mmci_host {
unsigned
int
oldstat
;
unsigned
int
oldstat
;
/* pio stuff */
/* pio stuff */
void
*
buffer
;
unsigned
int
offset
;
unsigned
int
size
;
unsigned
int
size
;
};
};
#define to_mmci_host(mmc) container_of(mmc, struct mmci_host, mmc)
drivers/mmc/pxamci.c
View file @
88e1f06c
...
@@ -530,6 +530,7 @@ static int pxamci_remove(struct device *dev)
...
@@ -530,6 +530,7 @@ static int pxamci_remove(struct device *dev)
return
0
;
return
0
;
}
}
#ifdef CONFIG_PM
static
int
pxamci_suspend
(
struct
device
*
dev
,
u32
state
,
u32
level
)
static
int
pxamci_suspend
(
struct
device
*
dev
,
u32
state
,
u32
level
)
{
{
struct
mmc_host
*
mmc
=
dev_get_drvdata
(
dev
);
struct
mmc_host
*
mmc
=
dev_get_drvdata
(
dev
);
...
@@ -551,6 +552,10 @@ static int pxamci_resume(struct device *dev, u32 level)
...
@@ -551,6 +552,10 @@ static int pxamci_resume(struct device *dev, u32 level)
return
ret
;
return
ret
;
}
}
#else
#define pxamci_suspend NULL
#define pxamci_resume NULL
#endif
static
struct
device_driver
pxamci_driver
=
{
static
struct
device_driver
pxamci_driver
=
{
.
name
=
"pxa2xx-mci"
,
.
name
=
"pxa2xx-mci"
,
...
...
include/linux/mmc/host.h
View file @
88e1f06c
...
@@ -64,12 +64,18 @@ struct device;
...
@@ -64,12 +64,18 @@ struct device;
struct
mmc_host
{
struct
mmc_host
{
struct
device
*
dev
;
struct
device
*
dev
;
struct
mmc_host_ops
*
ops
;
struct
mmc_host_ops
*
ops
;
void
*
priv
;
unsigned
int
f_min
;
unsigned
int
f_min
;
unsigned
int
f_max
;
unsigned
int
f_max
;
u32
ocr_avail
;
u32
ocr_avail
;
char
host_name
[
8
];
char
host_name
[
8
];
/* host specific block data */
unsigned
int
max_seg_size
;
/* see blk_queue_max_segment_size */
unsigned
short
max_hw_segs
;
/* see blk_queue_max_hw_segments */
unsigned
short
max_phys_segs
;
/* see blk_queue_max_phys_segments */
unsigned
short
max_sectors
;
/* see blk_queue_max_sectors */
unsigned
short
unused
;
/* private data */
/* private data */
struct
mmc_ios
ios
;
/* current io bus settings */
struct
mmc_ios
ios
;
/* current io bus settings */
u32
ocr
;
/* the current OCR setting */
u32
ocr
;
/* the current OCR setting */
...
...
include/linux/mmc/mmc.h
View file @
88e1f06c
...
@@ -28,6 +28,16 @@ struct mmc_command {
...
@@ -28,6 +28,16 @@ struct mmc_command {
#define MMC_RSP_CRC (1 << 3)
/* expect valid crc */
#define MMC_RSP_CRC (1 << 3)
/* expect valid crc */
#define MMC_RSP_BUSY (1 << 4)
/* card may send busy */
#define MMC_RSP_BUSY (1 << 4)
/* card may send busy */
/*
* These are the response types, and correspond to valid bit
* patterns of the above flags. One additional valid pattern
* is all zeros, which means we don't expect a response.
*/
#define MMC_RSP_R1 (MMC_RSP_SHORT|MMC_RSP_CRC)
#define MMC_RSP_R1B (MMC_RSP_SHORT|MMC_RSP_CRC|MMC_RSP_BUSY)
#define MMC_RSP_R2 (MMC_RSP_LONG|MMC_RSP_CRC)
#define MMC_RSP_R3 (MMC_RSP_SHORT)
unsigned
int
retries
;
/* max number of retries */
unsigned
int
retries
;
/* max number of retries */
unsigned
int
error
;
/* command error */
unsigned
int
error
;
/* command error */
...
...
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