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
34f16cd8
Commit
34f16cd8
authored
Apr 26, 2002
by
Anton Blanchard
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
ppc64: pci_dma updates - remove excess syncs and general cleanup, from
Mike Corrigan
parent
5b78e64f
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
288 additions
and
295 deletions
+288
-295
arch/ppc64/kernel/pSeries_lpar.c
arch/ppc64/kernel/pSeries_lpar.c
+20
-75
arch/ppc64/kernel/pci.c
arch/ppc64/kernel/pci.c
+2
-0
arch/ppc64/kernel/pci_dma.c
arch/ppc64/kernel/pci_dma.c
+261
-215
include/asm-ppc64/machdep.h
include/asm-ppc64/machdep.h
+2
-4
include/asm-ppc64/pci_dma.h
include/asm-ppc64/pci_dma.h
+3
-1
No files found.
arch/ppc64/kernel/pSeries_lpar.c
View file @
34f16cd8
...
@@ -219,7 +219,7 @@ long plpar_xirr(unsigned long *xirr_ret)
...
@@ -219,7 +219,7 @@ long plpar_xirr(unsigned long *xirr_ret)
static
void
tce_build_pSeriesLP
(
struct
TceTable
*
tbl
,
long
tcenum
,
static
void
tce_build_pSeriesLP
(
struct
TceTable
*
tbl
,
long
tcenum
,
unsigned
long
uaddr
,
int
direction
)
unsigned
long
uaddr
,
int
direction
)
{
{
u64
set
TceR
c
;
u64
set
_tce_r
c
;
union
Tce
tce
;
union
Tce
tce
;
PPCDBG
(
PPCDBG_TCE
,
"build_tce: uaddr = 0x%lx
\n
"
,
uaddr
);
PPCDBG
(
PPCDBG_TCE
,
"build_tce: uaddr = 0x%lx
\n
"
,
uaddr
);
...
@@ -232,90 +232,35 @@ static void tce_build_pSeriesLP(struct TceTable *tbl, long tcenum,
...
@@ -232,90 +232,35 @@ static void tce_build_pSeriesLP(struct TceTable *tbl, long tcenum,
tce
.
tceBits
.
readWrite
=
1
;
tce
.
tceBits
.
readWrite
=
1
;
if
(
direction
!=
PCI_DMA_TODEVICE
)
tce
.
tceBits
.
pciWrite
=
1
;
if
(
direction
!=
PCI_DMA_TODEVICE
)
tce
.
tceBits
.
pciWrite
=
1
;
set
TceR
c
=
plpar_tce_put
((
u64
)
tbl
->
index
,
set
_tce_r
c
=
plpar_tce_put
((
u64
)
tbl
->
index
,
(
u64
)
tcenum
<<
12
,
(
u64
)
tcenum
<<
12
,
tce
.
wholeTce
);
tce
.
wholeTce
);
/* Make sure the update is visible to hardware.
* ToDo: sync after setting *all* the tce's.
*/
__asm__
__volatile__
(
"sync"
:
:
:
"memory"
);
if
(
set
TceR
c
)
{
if
(
set
_tce_r
c
)
{
PPCDBG
(
PPCDBG_TCE
,
"setTce failed. rc=%ld
\n
"
,
setTceR
c
);
printk
(
"tce_build_pSeriesLP: plpar_tce_put failed. rc=%ld
\n
"
,
set_tce_r
c
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
index = 0x%lx
\n
"
,
(
u64
)
tbl
->
index
);
printk
(
"
\t
index = 0x%lx
\n
"
,
(
u64
)
tbl
->
index
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
tcenum = 0x%lx
\n
"
,
(
u64
)
tcenum
);
printk
(
"
\t
tcenum = 0x%lx
\n
"
,
(
u64
)
tcenum
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
tce val = 0x%lx
\n
"
,
tce
.
wholeTce
);
printk
(
"
\t
tce val = 0x%lx
\n
"
,
tce
.
wholeTce
);
}
}
}
}
static
inline
void
free_tce_range
(
struct
TceTable
*
tbl
,
static
void
tce_free_one_pSeriesLP
(
struct
TceTable
*
tbl
,
long
tcenum
)
long
tcenum
,
unsigned
order
)
{
unsigned
long
flags
;
/* Lock the tce allocation bitmap */
spin_lock_irqsave
(
&
(
tbl
->
lock
),
flags
);
/* Do the actual work */
free_tce_range_nolock
(
tbl
,
tcenum
,
order
);
/* Unlock the tce allocation bitmap */
spin_unlock_irqrestore
(
&
(
tbl
->
lock
),
flags
);
}
static
void
tce_free_pSeriesLP
(
struct
TceTable
*
tbl
,
dma_addr_t
dma_addr
,
unsigned
order
,
unsigned
numPages
)
{
{
u64
setTceRc
;
u64
set_tce_rc
;
long
tcenum
,
freeTce
,
maxTcenum
;
unsigned
i
;
union
Tce
tce
;
union
Tce
tce
;
maxTcenum
=
(
tbl
->
size
*
(
PAGE_SIZE
/
sizeof
(
union
Tce
)))
-
1
;
tce
.
wholeTce
=
0
;
set_tce_rc
=
plpar_tce_put
((
u64
)
tbl
->
index
,
tcenum
=
dma_addr
>>
PAGE_SHIFT
;
(
u64
)
tcenum
<<
12
,
tce
.
wholeTce
);
freeTce
=
tcenum
-
tbl
->
startOffset
;
if
(
set_tce_rc
)
{
printk
(
"tce_free_one_pSeriesLP: plpar_tce_put failed
\n
"
);
if
(
freeTce
>
maxTcenum
)
{
printk
(
"
\t
rc = %ld
\n
"
,
set_tce_rc
);
printk
(
"free_tces: tcenum > maxTcenum
\n
"
);
printk
(
"
\t
index = 0x%lx
\n
"
,
(
u64
)
tbl
->
index
);
printk
(
"
\t
tcenum = 0x%lx
\n
"
,
tcenum
);
printk
(
"
\t
tcenum = 0x%lx
\n
"
,
(
u64
)
tcenum
);
printk
(
"
\t
freeTce = 0x%lx
\n
"
,
freeTce
);
printk
(
"
\t
tce val = 0x%lx
\n
"
,
tce
.
wholeTce
);
printk
(
"
\t
maxTcenum = 0x%lx
\n
"
,
maxTcenum
);
printk
(
"
\t
TCE Table = 0x%lx
\n
"
,
(
u64
)
tbl
);
printk
(
"
\t
bus# = 0x%lx
\n
"
,
(
u64
)
tbl
->
busNumber
);
printk
(
"
\t
size = 0x%lx
\n
"
,
(
u64
)
tbl
->
size
);
printk
(
"
\t
startOff = 0x%lx
\n
"
,
(
u64
)
tbl
->
startOffset
);
printk
(
"
\t
index = 0x%lx
\n
"
,
(
u64
)
tbl
->
index
);
return
;
}
for
(
i
=
0
;
i
<
numPages
;
++
i
)
{
tce
.
wholeTce
=
0
;
setTceRc
=
plpar_tce_put
((
u64
)
tbl
->
index
,
(
u64
)
tcenum
<<
12
,
/* note: not freeTce */
tce
.
wholeTce
);
if
(
setTceRc
)
{
printk
(
"tce_free: setTce failed
\n
"
);
printk
(
"
\t
rc = %ld
\n
"
,
setTceRc
);
printk
(
"
\t
index = 0x%lx
\n
"
,
(
u64
)
tbl
->
index
);
printk
(
"
\t
tcenum = 0x%lx
\n
"
,
(
u64
)
tcenum
);
printk
(
"
\t
freeTce = 0x%lx
\n
"
,
(
u64
)
freeTce
);
printk
(
"
\t
tce val = 0x%lx
\n
"
,
tce
.
wholeTce
);
}
++
tcenum
;
}
}
/* Make sure the update is visible to hardware. */
__asm__
__volatile__
(
"sync"
:
:
:
"memory"
);
free_tce_range
(
tbl
,
freeTce
,
order
);
}
}
/* PowerPC Interrupts for lpar. */
/* PowerPC Interrupts for lpar. */
...
@@ -446,7 +391,7 @@ void pSeriesLP_init_early(void)
...
@@ -446,7 +391,7 @@ void pSeriesLP_init_early(void)
pSeries_lpar_mm_init
();
pSeries_lpar_mm_init
();
ppc_md
.
tce_build
=
tce_build_pSeriesLP
;
ppc_md
.
tce_build
=
tce_build_pSeriesLP
;
ppc_md
.
tce_free
=
tce_fre
e_pSeriesLP
;
ppc_md
.
tce_free
_one
=
tce_free_on
e_pSeriesLP
;
#ifdef CONFIG_SMP
#ifdef CONFIG_SMP
smp_init_pSeries
();
smp_init_pSeries
();
...
...
arch/ppc64/kernel/pci.c
View file @
34f16cd8
...
@@ -63,6 +63,8 @@ int pci_assign_all_busses = 0;
...
@@ -63,6 +63,8 @@ int pci_assign_all_busses = 0;
struct
pci_controller
*
hose_head
;
struct
pci_controller
*
hose_head
;
struct
pci_controller
**
hose_tail
=
&
hose_head
;
struct
pci_controller
**
hose_tail
=
&
hose_head
;
LIST_HEAD
(
iSeries_Global_Device_List
);
/*******************************************************************
/*******************************************************************
* Counters and control flags.
* Counters and control flags.
*******************************************************************/
*******************************************************************/
...
...
arch/ppc64/kernel/pci_dma.c
View file @
34f16cd8
...
@@ -43,7 +43,9 @@
...
@@ -43,7 +43,9 @@
#include "pci.h"
#include "pci.h"
// #define DEBUG_TCE 1
/* #define DEBUG_TCE 1 */
/* #define MONITOR_TCE 1 */
/* Turn on to sanity check TCE generation. */
/* Initialize so this guy does not end up in the BSS section.
/* Initialize so this guy does not end up in the BSS section.
* Only used to pass OF initialization data set in prom.c into the main
* Only used to pass OF initialization data set in prom.c into the main
...
@@ -53,12 +55,13 @@ extern struct _of_tce_table of_tce_table[];
...
@@ -53,12 +55,13 @@ extern struct _of_tce_table of_tce_table[];
extern
struct
pci_controller
*
hose_head
;
extern
struct
pci_controller
*
hose_head
;
extern
struct
pci_controller
**
hose_tail
;
extern
struct
pci_controller
**
hose_tail
;
extern
struct
list_head
iSeries_Global_Device_List
;
struct
TceTable
virtBusVethTceTable
;
/* Tce table for virtual ethernet */
struct
TceTable
virtBusVethTceTable
;
/* Tce table for virtual ethernet */
struct
TceTable
virtBusVioTceTable
;
/* Tce table for virtual I/O */
struct
TceTable
virtBusVioTceTable
;
/* Tce table for virtual I/O */
struct
device_node
iSeries_veth_dev_node
=
{
tce_t
able
:
&
virtBusVethTceTable
};
struct
iSeries_Device_Node
iSeries_veth_dev_node
=
{
LogicalSlot
:
0xFF
,
DevTceT
able
:
&
virtBusVethTceTable
};
struct
device_node
iSeries_vio_dev_node
=
{
tce_t
able
:
&
virtBusVioTceTable
};
struct
iSeries_Device_Node
iSeries_vio_dev_node
=
{
LogicalSlot
:
0xFF
,
DevTceT
able
:
&
virtBusVioTceTable
};
struct
pci_dev
iSeries_veth_dev_st
=
{
sysdata
:
&
iSeries_veth_dev_node
};
struct
pci_dev
iSeries_veth_dev_st
=
{
sysdata
:
&
iSeries_veth_dev_node
};
struct
pci_dev
iSeries_vio_dev_st
=
{
sysdata
:
&
iSeries_vio_dev_node
};
struct
pci_dev
iSeries_vio_dev_st
=
{
sysdata
:
&
iSeries_vio_dev_node
};
...
@@ -66,7 +69,8 @@ struct pci_dev iSeries_vio_dev_st = { sysdata: &iSeries_vio_dev_node };
...
@@ -66,7 +69,8 @@ struct pci_dev iSeries_vio_dev_st = { sysdata: &iSeries_vio_dev_node };
struct
pci_dev
*
iSeries_veth_dev
=
&
iSeries_veth_dev_st
;
struct
pci_dev
*
iSeries_veth_dev
=
&
iSeries_veth_dev_st
;
struct
pci_dev
*
iSeries_vio_dev
=
&
iSeries_vio_dev_st
;
struct
pci_dev
*
iSeries_vio_dev
=
&
iSeries_vio_dev_st
;
struct
TceTable
*
tceTables
[
256
];
/* Tce tables for 256 busses
/* Device TceTable is stored in Device Node */
/* struct TceTable * tceTables[256]; */
/* Tce tables for 256 busses
* Bus 255 is the virtual bus
* Bus 255 is the virtual bus
* zero indicates no bus defined
* zero indicates no bus defined
*/
*/
...
@@ -116,6 +120,9 @@ static dma_addr_t create_tces_sg( struct TceTable *tbl,
...
@@ -116,6 +120,9 @@ static dma_addr_t create_tces_sg( struct TceTable *tbl,
unsigned
numTces
,
unsigned
numTces
,
int
direction
);
int
direction
);
static
void
getTceTableParmsiSeries
(
struct
iSeries_Device_Node
*
DevNode
,
struct
TceTable
*
tce_table_parms
);
static
void
getTceTableParmsPSeries
(
struct
pci_controller
*
phb
,
static
void
getTceTableParmsPSeries
(
struct
pci_controller
*
phb
,
struct
device_node
*
dn
,
struct
device_node
*
dn
,
struct
TceTable
*
tce_table_parms
);
struct
TceTable
*
tce_table_parms
);
...
@@ -124,6 +131,8 @@ static void getTceTableParmsPSeriesLP(struct pci_controller *phb,
...
@@ -124,6 +131,8 @@ static void getTceTableParmsPSeriesLP(struct pci_controller *phb,
struct
device_node
*
dn
,
struct
device_node
*
dn
,
struct
TceTable
*
newTceTable
);
struct
TceTable
*
newTceTable
);
static
struct
TceTable
*
findHwTceTable
(
struct
TceTable
*
newTceTable
);
void
create_pci_bus_tce_table
(
unsigned
long
token
);
void
create_pci_bus_tce_table
(
unsigned
long
token
);
u8
iSeries_Get_Bus
(
struct
pci_dev
*
dv
)
u8
iSeries_Get_Bus
(
struct
pci_dev
*
dv
)
...
@@ -137,11 +146,11 @@ static inline struct TceTable *get_tce_table(struct pci_dev *dev)
...
@@ -137,11 +146,11 @@ static inline struct TceTable *get_tce_table(struct pci_dev *dev)
dev
=
ppc64_isabridge_dev
;
dev
=
ppc64_isabridge_dev
;
if
(
!
dev
)
if
(
!
dev
)
return
NULL
;
return
NULL
;
if
(
(
naca
->
platform
==
PLATFORM_ISERIES_LPAR
)
&&
(
dev
->
bus
)
)
if
(
naca
->
platform
==
PLATFORM_ISERIES_LPAR
)
{
return
tceTables
[
dev
->
bus
->
number
]
;
return
ISERIES_DEVNODE
(
dev
)
->
DevTceTable
;
/* On the iSeries, the virtual bus will take this path. There is a */
}
else
{
/* fake pci_dev and dev_node built and used. */
return
PCI_GET_DN
(
dev
)
->
tce_table
;
return
PCI_GET_DN
(
dev
)
->
tce_table
;
}
}
}
static
unsigned
long
__inline__
count_leading_zeros64
(
unsigned
long
x
)
static
unsigned
long
__inline__
count_leading_zeros64
(
unsigned
long
x
)
...
@@ -180,13 +189,8 @@ static void tce_build_iSeries(struct TceTable *tbl, long tcenum,
...
@@ -180,13 +189,8 @@ static void tce_build_iSeries(struct TceTable *tbl, long tcenum,
setTceRc
=
HvCallXm_setTce
((
u64
)
tbl
->
index
,
setTceRc
=
HvCallXm_setTce
((
u64
)
tbl
->
index
,
(
u64
)
tcenum
,
(
u64
)
tcenum
,
tce
.
wholeTce
);
tce
.
wholeTce
);
if
(
setTceRc
)
{
if
(
setTceRc
)
{
printk
(
"PCI: tce_build failed 0x%lx tcenum: 0x%lx
\n
"
,
setTceRc
,
(
u64
)
tcenum
);
panic
(
"PCI_DMA: HvCallXm_setTce failed, Rc: 0x%lx
\n
"
,
setTceRc
);
//PPCDBG(PPCDBG_TCE, "setTce failed. rc=%ld\n", setTceRc);
//PPCDBG(PPCDBG_TCE, "\tindex = 0x%lx\n", (u64)tbl->index);
//PPCDBG(PPCDBG_TCE, "\ttce num = 0x%lx\n", (u64)tcenum);
//PPCDBG(PPCDBG_TCE, "\ttce val = 0x%lx\n", tce.wholeTce );
}
}
}
}
...
@@ -209,8 +213,6 @@ static void tce_build_pSeries(struct TceTable *tbl, long tcenum,
...
@@ -209,8 +213,6 @@ static void tce_build_pSeries(struct TceTable *tbl, long tcenum,
tce_addr
=
((
union
Tce
*
)
tbl
->
base
)
+
tcenum
;
tce_addr
=
((
union
Tce
*
)
tbl
->
base
)
+
tcenum
;
*
tce_addr
=
(
union
Tce
)
tce
.
wholeTce
;
*
tce_addr
=
(
union
Tce
)
tce
.
wholeTce
;
/* Make sure the update is visible to hardware. */
__asm__
__volatile__
(
"sync"
:
:
:
"memory"
);
}
}
/*
/*
...
@@ -245,18 +247,18 @@ static struct TceTable *build_tce_table( struct TceTable * tbl )
...
@@ -245,18 +247,18 @@ static struct TceTable *build_tce_table( struct TceTable * tbl )
}
}
PPCDBG
(
PPCDBG_TCEINIT
,
"build_tce_table: totalBytes=%ld
\n
"
,
totalBytes
);
PPCDBG
(
PPCDBG_TCEINIT
,
"build_tce_table: totalBytes=%ld
\n
"
,
totalBytes
);
pos
=
(
char
*
)
__get_free_pages
(
GFP_ATOMIC
,
get_order
(
totalBytes
));
pos
=
(
char
*
)
__get_free_pages
(
GFP_ATOMIC
,
get_order
(
totalBytes
));
if
(
!
pos
)
return
NULL
;
if
(
pos
==
NULL
)
{
panic
(
"PCI_DMA: Allocation failed in build_tce_table!
\n
"
);
}
memset
(
pos
,
0
,
totalBytes
);
/* For each level, fill in the pointer to the bit map,
/* For each level, fill in the pointer to the bit map,
* and turn on the last bit in the bit map (if the
* and turn on the last bit in the bit map (if the
* number of bits in the map is odd). The highest
* number of bits in the map is odd). The highest
* level will get all of its bits turned on.
* level will get all of its bits turned on.
*/
*/
memset
(
pos
,
0
,
totalBytes
);
for
(
i
=
0
;
i
<
NUM_TCE_LEVELS
;
++
i
)
{
for
(
i
=
0
;
i
<
NUM_TCE_LEVELS
;
++
i
)
{
if
(
numBytes
[
i
]
)
{
if
(
numBytes
[
i
]
)
{
tbl
->
mlbm
.
level
[
i
].
map
=
pos
;
tbl
->
mlbm
.
level
[
i
].
map
=
pos
;
...
@@ -333,9 +335,9 @@ static long alloc_tce_range_nolock( struct TceTable *tbl, unsigned order )
...
@@ -333,9 +335,9 @@ static long alloc_tce_range_nolock( struct TceTable *tbl, unsigned order )
* biggest, indicate failure
* biggest, indicate failure
*/
*/
if
(
order
>=
NUM_TCE_LEVELS
)
{
if
(
order
>=
NUM_TCE_LEVELS
)
{
PPCDBG
(
PPCDBG_TCE
,
/* This can happen if block of TCE's are not found. This code */
"alloc_tce_range_nolock: invalid order: %d
\n
"
,
order
);
/* maybe in a recursive loop looking up the bit map for the range.*/
return
-
1
;
panic
(
"PCI_DMA: alloc_tce_range_nolock: invalid order: %d
\n
"
,
order
)
;
}
}
numBits
=
tbl
->
mlbm
.
level
[
order
].
numBits
;
numBits
=
tbl
->
mlbm
.
level
[
order
].
numBits
;
...
@@ -352,7 +354,7 @@ static long alloc_tce_range_nolock( struct TceTable *tbl, unsigned order )
...
@@ -352,7 +354,7 @@ static long alloc_tce_range_nolock( struct TceTable *tbl, unsigned order )
* number (of this size)
* number (of this size)
*/
*/
bit
=
count_leading_zeros64
(
*
map
);
bit
=
count_leading_zeros64
(
*
map
);
block
=
(
i
*
64
)
+
bit
;
block
=
(
i
*
64
)
+
bit
;
/* Bit count to free entry */
/* turn off the bit in the map to indicate
/* turn off the bit in the map to indicate
* that the block is now in use
* that the block is now in use
...
@@ -373,10 +375,12 @@ static long alloc_tce_range_nolock( struct TceTable *tbl, unsigned order )
...
@@ -373,10 +375,12 @@ static long alloc_tce_range_nolock( struct TceTable *tbl, unsigned order )
#ifdef DEBUG_TCE
#ifdef DEBUG_TCE
if
(
tcenum
==
-
1
)
{
if
(
tcenum
==
-
1
)
{
PPCDBG
(
PPCDBG_TCE
,
"alloc_tce_range_nolock: no available blocks of order = %d
\n
"
,
order
);
PPCDBG
(
PPCDBG_TCE
,
"alloc_tce_range_nolock: no available blocks of order = %d
\n
"
,
order
);
if
(
order
<
tbl
->
mlbm
.
maxLevel
)
if
(
order
<
tbl
->
mlbm
.
maxLevel
)
{
PPCDBG
(
PPCDBG_TCE
,
"alloc_tce_range_nolock: trying next bigger size
\n
"
);
PPCDBG
(
PPCDBG_TCE
,
"alloc_tce_range_nolock: trying next bigger size
\n
"
);
else
}
PPCDBG
(
PPCDBG_TCE
,
"alloc_tce_range_nolock: maximum size reached...failing
\n
"
);
else
{
panic
(
"PCI_DMA: alloc_tce_range_nolock: maximum size reached...failing
\n
"
);
}
}
}
#endif
#endif
...
@@ -395,7 +399,6 @@ static long alloc_tce_range_nolock( struct TceTable *tbl, unsigned order )
...
@@ -395,7 +399,6 @@ static long alloc_tce_range_nolock( struct TceTable *tbl, unsigned order )
* (or -1 if we failed)
* (or -1 if we failed)
*/
*/
return
tcenum
;
return
tcenum
;
}
}
static
inline
void
free_tce_range
(
struct
TceTable
*
tbl
,
static
inline
void
free_tce_range
(
struct
TceTable
*
tbl
,
...
@@ -422,37 +425,25 @@ void free_tce_range_nolock(struct TceTable *tbl,
...
@@ -422,37 +425,25 @@ void free_tce_range_nolock(struct TceTable *tbl,
unsigned
char
*
map
,
*
bytep
;
unsigned
char
*
map
,
*
bytep
;
if
(
order
>=
NUM_TCE_LEVELS
)
{
if
(
order
>=
NUM_TCE_LEVELS
)
{
PPCDBG
(
PPCDBG_TCE
,
panic
(
"PCI_DMA: free_tce_range: invalid order: 0x%x
\n
"
,
order
);
"free_tce_range: invalid order: %d, tcenum = %d
\n
"
,
order
,
tcenum
);
return
;
return
;
}
}
block
=
tcenum
>>
order
;
block
=
tcenum
>>
order
;
#ifdef
DEBUG
_TCE
#ifdef
MONITOR
_TCE
if
(
tcenum
!=
(
block
<<
order
)
)
{
if
(
tcenum
!=
(
block
<<
order
)
)
{
PPCDBG
(
PPCDBG_TCE
,
printk
(
"PCI_DMA: Free_tce_range: tcenum %lx misaligned for order %x
\n
"
,
tcenum
,
order
);
"free_tce_range: tcenum %lx misaligned for order %x
\n
"
,
tcenum
,
order
);
return
;
return
;
}
}
if
(
block
>=
tbl
->
mlbm
.
level
[
order
].
numBits
)
{
if
(
block
>=
tbl
->
mlbm
.
level
[
order
].
numBits
)
{
PPCDBG
(
PPCDBG_TCE
,
printk
(
"PCI_DMA: Free_tce_range: tcenum %lx is outside the range of this map (order %x, numBits %lx
\n
"
,
"free_tce_range: tcenum %lx is outside the range of this map (order %x, numBits %lx
\n
"
,
tcenum
,
order
,
tbl
->
mlbm
.
level
[
order
].
numBits
);
tcenum
,
order
,
tbl
->
mlbm
.
level
[
order
].
numBits
);
return
;
return
;
}
}
if
(
test_tce_range
(
tbl
,
tcenum
,
order
)
)
{
if
(
test_tce_range
(
tbl
,
tcenum
,
order
)
)
{
PPCDBG
(
PPCDBG_TCE
,
printk
(
"PCI_DMA: Freeing range not allocated: tTceTable %p, tcenum %lx, order %x
\n
"
,
tbl
,
tcenum
,
order
);
"free_tce_range: freeing range not allocated.
\n
"
);
return
;
PPCDBG
(
PPCDBG_TCE
,
"
\t
TceTable %p, tcenum %lx, order %x
\n
"
,
tbl
,
tcenum
,
order
);
}
}
#endif
#endif
...
@@ -463,13 +454,14 @@ void free_tce_range_nolock(struct TceTable *tbl,
...
@@ -463,13 +454,14 @@ void free_tce_range_nolock(struct TceTable *tbl,
bytep
=
map
+
byte
;
bytep
=
map
+
byte
;
#ifdef DEBUG_TCE
#ifdef DEBUG_TCE
PPCDBG
(
PPCDBG_TCE
,
PPCDBG
(
PPCDBG_TCE
,
"free_tce_range_nolock: freeing block %ld (byte=%d, bit=%d) of order %d
\n
"
,
"free_tce_range_nolock: freeing block %ld (byte=%d, bit=%d) of order %d
\n
"
,
block
,
byte
,
bit
,
order
);
block
,
byte
,
bit
,
order
);
if
(
*
bytep
&
mask
)
#endif
PPCDBG
(
PPCDBG_TCE
,
"free_tce_range: already free: TceTable %p, tcenum %lx, order %x
\n
"
,
#ifdef MONITOR_TCE
tbl
,
tcenum
,
order
);
if
(
*
bytep
&
mask
)
{
panic
(
"PCI_DMA: Tce already free: TceTable %p, tcenum %lx, order %x
\n
"
,
tbl
,
tcenum
,
order
);
}
#endif
#endif
*
bytep
|=
mask
;
*
bytep
|=
mask
;
...
@@ -481,7 +473,8 @@ void free_tce_range_nolock(struct TceTable *tbl,
...
@@ -481,7 +473,8 @@ void free_tce_range_nolock(struct TceTable *tbl,
* we are freeing the last block we can't buddy up
* we are freeing the last block we can't buddy up
* Don't buddy up if it's in the first 1/4 of the level
* Don't buddy up if it's in the first 1/4 of the level
*/
*/
if
((
block
>
(
tbl
->
mlbm
.
level
[
order
].
numBits
/
4
)
)
&&
if
((
order
<
tbl
->
mlbm
.
maxLevel
)
&&
(
block
>
(
tbl
->
mlbm
.
level
[
order
].
numBits
/
4
)
)
&&
((
block
<
tbl
->
mlbm
.
level
[
order
].
numBits
-
1
)
||
((
block
<
tbl
->
mlbm
.
level
[
order
].
numBits
-
1
)
||
(
0
==
(
tbl
->
mlbm
.
level
[
order
].
numBits
&
1
))))
{
(
0
==
(
tbl
->
mlbm
.
level
[
order
].
numBits
&
1
))))
{
/* See if we can buddy up the block we just freed */
/* See if we can buddy up the block we just freed */
...
@@ -555,101 +548,78 @@ static inline dma_addr_t get_tces( struct TceTable *tbl, unsigned order, void *p
...
@@ -555,101 +548,78 @@ static inline dma_addr_t get_tces( struct TceTable *tbl, unsigned order, void *p
++
tcenum
;
++
tcenum
;
uaddr
+=
PAGE_SIZE
;
uaddr
+=
PAGE_SIZE
;
}
}
/* Make sure the update is visible to hardware.
sync required to synchronize the update to
the TCE table with the MMIO that will send
the bus address to the IOA */
__asm__
__volatile__
(
"sync"
:
:
:
"memory"
);
}
}
else
else
{
PPCDBG
(
PPCDBG_TCE
,
"alloc_tce_range failed
\n
"
);
panic
(
"PCI_DMA: Tce Allocation failure in get_tces. 0x%p
\n
"
,
tbl
);
}
return
retTce
;
return
retTce
;
}
}
static
void
tce_free_iSeries
(
struct
TceTable
*
tbl
,
dma_addr_t
dma_addr
,
static
void
tce_free_one_iSeries
(
struct
TceTable
*
tbl
,
long
tcenum
)
unsigned
order
,
unsigned
numPages
)
{
{
u64
setTceRc
;
u64
set_tce_rc
;
long
tcenum
,
freeTce
,
maxTcenum
;
unsigned
i
;
union
Tce
tce
;
union
Tce
tce
;
tce
.
wholeTce
=
0
;
set_tce_rc
=
HvCallXm_setTce
((
u64
)
tbl
->
index
,
(
u64
)
tcenum
,
tce
.
wholeTce
);
if
(
set_tce_rc
)
panic
(
"PCI_DMA: HvCallXm_setTce failed, Rc: 0x%lx
\n
"
,
set_tce_rc
);
maxTcenum
=
(
tbl
->
size
*
(
PAGE_SIZE
/
sizeof
(
union
Tce
)))
-
1
;
}
tcenum
=
dma_addr
>>
PAGE_SHIFT
;
freeTce
=
tcenum
-
tbl
->
startOffset
;
static
void
tce_free_one_pSeries
(
struct
TceTable
*
tbl
,
long
tcenum
)
{
if
(
freeTce
>
maxTcenum
)
{
union
Tce
tce
;
PPCDBG
(
PPCDBG_TCE
,
"free_tces: tcenum > maxTcenum
\n
"
);
union
Tce
*
tce_addr
;
PPCDBG
(
PPCDBG_TCE
,
"
\t
tcenum = 0x%lx
\n
"
,
tcenum
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
maxTcenum = 0x%lx
\n
"
,
maxTcenum
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
TCE Table = 0x%lx
\n
"
,
(
u64
)
tbl
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
bus# = 0x%lx
\n
"
,
(
u64
)
tbl
->
busNumber
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
size = 0x%lx
\n
"
,
(
u64
)
tbl
->
size
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
startOff = 0x%lx
\n
"
,
(
u64
)
tbl
->
startOffset
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
index = 0x%lx
\n
"
,
(
u64
)
tbl
->
index
);
return
;
}
for
(
i
=
0
;
i
<
numPages
;
++
i
)
{
tce
.
wholeTce
=
0
;
setTceRc
=
HvCallXm_setTce
((
u64
)
tbl
->
index
,
(
u64
)
tcenum
,
tce
.
wholeTce
);
if
(
setTceRc
)
{
printk
(
"PCI: tce_free failed 0x%lx tcenum: 0x%lx
\n
"
,
setTceRc
,
(
u64
)
tcenum
);
//PPCDBG(PPCDBG_TCE, "tce_free: setTce failed\n");
//PPCDBG(PPCDBG_TCE, "\trc = 0x%lx\n", setTceRc);
//PPCDBG(PPCDBG_TCE, "\tindex = 0x%lx\n", (u64)tbl->index);
//PPCDBG(PPCDBG_TCE, "\ttce num = 0x%lx\n", (u64)tcenum);
//PPCDBG(PPCDBG_TCE, "\ttce val = 0x%lx\n", tce.wholeTce );
}
++
tcenum
;
tce
.
wholeTce
=
0
;
}
tce_addr
=
((
union
Tce
*
)
tbl
->
base
)
+
tcenum
;
*
tce_addr
=
(
union
Tce
)
tce
.
wholeTce
;
free_tce_range
(
tbl
,
freeTce
,
order
);
}
}
static
void
tce_free
_pSeries
(
struct
TceTable
*
tbl
,
dma_addr_t
dma_addr
,
static
void
tce_free
(
struct
TceTable
*
tbl
,
dma_addr_t
dma_addr
,
unsigned
order
,
unsigned
num
P
ages
)
unsigned
order
,
unsigned
num
_p
ages
)
{
{
long
tcenum
,
freeTce
,
maxTcenum
;
long
tcenum
,
total_tces
,
free_tce
;
unsigned
i
;
unsigned
i
;
union
Tce
tce
;
union
Tce
*
tce_addr
;
maxTcenum
=
(
tbl
->
size
*
(
PAGE_SIZE
/
sizeof
(
union
Tce
)))
-
1
;
total_tces
=
(
tbl
->
size
*
(
PAGE_SIZE
/
sizeof
(
union
Tce
)))
;
tcenum
=
dma_addr
>>
PAGE_SHIFT
;
tcenum
=
dma_addr
>>
PAGE_SHIFT
;
// tcenum -= tbl->startOffset;
free_tce
=
tcenum
-
tbl
->
startOffset
;
freeTce
=
tcenum
-
tbl
->
startOffset
;
if
(
(
(
free_tce
+
num_pages
)
>
total_tces
)
||
(
tcenum
<
tbl
->
startOffset
)
)
{
if
(
freeTce
>
maxTcenum
)
{
printk
(
"tce_free: invalid tcenum
\n
"
);
PPCDBG
(
PPCDBG_TCE
,
"free_tces: tcenum > maxTcenum
\n
"
);
printk
(
"
\t
tcenum = 0x%lx
\n
"
,
tcenum
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
tcenum = 0x%lx
\n
"
,
tcenum
);
printk
(
"
\t
TCE Table = 0x%lx
\n
"
,
(
u64
)
tbl
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
maxTcenum = 0x%lx
\n
"
,
maxTcenum
);
printk
(
"
\t
bus# = 0x%lx
\n
"
,
(
u64
)
tbl
->
busNumber
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
TCE Table = 0x%lx
\n
"
,
(
u64
)
tbl
);
printk
(
"
\t
size = 0x%lx
\n
"
,
(
u64
)
tbl
->
size
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
bus# = 0x%lx
\n
"
,
printk
(
"
\t
startOff = 0x%lx
\n
"
,
(
u64
)
tbl
->
startOffset
);
(
u64
)
tbl
->
busNumber
);
printk
(
"
\t
index = 0x%lx
\n
"
,
(
u64
)
tbl
->
index
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
size = 0x%lx
\n
"
,
(
u64
)
tbl
->
size
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
startOff = 0x%lx
\n
"
,
(
u64
)
tbl
->
startOffset
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
index = 0x%lx
\n
"
,
(
u64
)
tbl
->
index
);
return
;
return
;
}
}
for
(
i
=
0
;
i
<
numPages
;
++
i
)
{
for
(
i
=
0
;
i
<
num_pages
;
++
i
)
{
tce
.
wholeTce
=
0
;
ppc_md
.
tce_free_one
(
tbl
,
tcenum
);
tce_addr
=
((
union
Tce
*
)
tbl
->
base
)
+
tcenum
;
*
tce_addr
=
(
union
Tce
)
tce
.
wholeTce
;
++
tcenum
;
++
tcenum
;
}
}
/* Make sure the update is visible to hardware. */
/* No sync (to make TCE change visible) is required here.
__asm__
__volatile__
(
"sync"
:
:
:
"memory"
);
The lwsync when acquiring the lock in free_tce_range
is sufficient to synchronize with the bitmap.
*/
free_tce_range
(
tbl
,
free
T
ce
,
order
);
free_tce_range
(
tbl
,
free
_t
ce
,
order
);
}
}
void
__init
create_virtual_bus_tce_table
(
void
)
void
__init
create_virtual_bus_tce_table
(
void
)
...
@@ -679,7 +649,8 @@ void __init create_virtual_bus_tce_table(void)
...
@@ -679,7 +649,8 @@ void __init create_virtual_bus_tce_table(void)
t
=
build_tce_table
(
&
virtBusVethTceTable
);
t
=
build_tce_table
(
&
virtBusVethTceTable
);
if
(
t
)
{
if
(
t
)
{
tceTables
[
255
]
=
t
;
/* tceTables[255] = t; */
//VirtBusVethTceTable = t;
printk
(
"Virtual Bus VETH TCE table built successfully.
\n
"
);
printk
(
"Virtual Bus VETH TCE table built successfully.
\n
"
);
printk
(
" TCE table size = %ld entries
\n
"
,
printk
(
" TCE table size = %ld entries
\n
"
,
(
unsigned
long
)
t
->
size
*
(
PAGE_SIZE
/
sizeof
(
union
Tce
))
);
(
unsigned
long
)
t
->
size
*
(
PAGE_SIZE
/
sizeof
(
union
Tce
))
);
...
@@ -688,11 +659,11 @@ void __init create_virtual_bus_tce_table(void)
...
@@ -688,11 +659,11 @@ void __init create_virtual_bus_tce_table(void)
printk
(
" TCE table start entry = 0x%lx
\n
"
,
printk
(
" TCE table start entry = 0x%lx
\n
"
,
(
unsigned
long
)
t
->
startOffset
);
(
unsigned
long
)
t
->
startOffset
);
}
}
else
else
printk
(
"Virtual Bus VETH TCE table failed.
\n
"
);
printk
(
"Virtual Bus VETH TCE table failed.
\n
"
);
t
=
build_tce_table
(
&
virtBusVioTceTable
);
t
=
build_tce_table
(
&
virtBusVioTceTable
);
if
(
t
)
{
if
(
t
)
{
//VirtBusVioTceTable = t;
printk
(
"Virtual Bus VIO TCE table built successfully.
\n
"
);
printk
(
"Virtual Bus VIO TCE table built successfully.
\n
"
);
printk
(
" TCE table size = %ld entries
\n
"
,
printk
(
" TCE table size = %ld entries
\n
"
,
(
unsigned
long
)
t
->
size
*
(
PAGE_SIZE
/
sizeof
(
union
Tce
))
);
(
unsigned
long
)
t
->
size
*
(
PAGE_SIZE
/
sizeof
(
union
Tce
))
);
...
@@ -701,8 +672,7 @@ void __init create_virtual_bus_tce_table(void)
...
@@ -701,8 +672,7 @@ void __init create_virtual_bus_tce_table(void)
printk
(
" TCE table start entry = 0x%lx
\n
"
,
printk
(
" TCE table start entry = 0x%lx
\n
"
,
(
unsigned
long
)
t
->
startOffset
);
(
unsigned
long
)
t
->
startOffset
);
}
}
else
else
printk
(
"Virtual Bus VIO TCE table failed.
\n
"
);
printk
(
"Virtual Bus VIO TCE table failed.
\n
"
);
}
}
void
create_tce_tables_for_buses
(
struct
list_head
*
bus_list
)
void
create_tce_tables_for_buses
(
struct
list_head
*
bus_list
)
...
@@ -765,11 +735,12 @@ void create_tce_tables(void) {
...
@@ -765,11 +735,12 @@ void create_tce_tables(void) {
struct
pci_dev
*
dev
;
struct
pci_dev
*
dev
;
struct
device_node
*
dn
,
*
mydn
;
struct
device_node
*
dn
,
*
mydn
;
if
(
naca
->
platform
==
PLATFORM_PSERIES_LPAR
)
if
(
naca
->
platform
==
PLATFORM_PSERIES_LPAR
)
{
create_tce_tables_for_busesLP
(
&
pci_root_buses
);
create_tce_tables_for_busesLP
(
&
pci_root_buses
);
else
}
create_tce_tables_for_buses
(
&
pci_root_buses
);
else
{
create_tce_tables_for_buses
(
&
pci_root_buses
);
}
/* Now copy the tce_table ptr from the bus devices down to every
/* Now copy the tce_table ptr from the bus devices down to every
* pci device_node. This means get_tce_table() won't need to search
* pci device_node. This means get_tce_table() won't need to search
* up the device tree to find it.
* up the device tree to find it.
...
@@ -784,77 +755,46 @@ void create_tce_tables(void) {
...
@@ -784,77 +755,46 @@ void create_tce_tables(void) {
}
}
}
}
/*
/*
* iSeries token =
busNumber
* iSeries token =
iSeries_device_Node*
* pSeries token = pci_controller*
* pSeries token = pci_controller*
*
*/
*/
void
create_pci_bus_tce_table
(
unsigned
long
token
)
{
void
create_pci_bus_tce_table
(
unsigned
long
token
)
{
struct
TceTable
*
builtTceTable
;
struct
TceTable
*
newTceTable
;
struct
TceTable
*
newTceTable
;
struct
TceTableManagerCB
pciBusTceTableParms
;
u64
parmsPtr
;
PPCDBG
(
PPCDBG_TCE
,
"Entering create_pci_bus_tce_table.
\n
"
);
PPCDBG
(
PPCDBG_TCE
,
"Entering create_pci_bus_tce_table.
\n
"
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
token = 0x%lx
\n
"
,
token
);
PPCDBG
(
PPCDBG_TCE
,
"
\t
token = 0x%lx
\n
"
,
token
);
newTceTable
=
kmalloc
(
sizeof
(
struct
TceTable
),
GFP_KERNEL
);
newTceTable
=
(
struct
TceTable
*
)
kmalloc
(
sizeof
(
struct
TceTable
),
GFP_KERNEL
);
/*****************************************************************/
/* For the iSeries machines, the HvTce Table can be one of three */
/* flavors, */
/* - Single bus TCE table, */
/* - Tce Table Share between buses, */
/* - Tce Table per logical slot. */
/*****************************************************************/
if
(
naca
->
platform
==
PLATFORM_ISERIES_LPAR
)
{
if
(
naca
->
platform
==
PLATFORM_ISERIES_LPAR
)
{
if
(
token
>
254
)
{
printk
(
"PCI: Bus TCE table failed, invalid bus number %lu
\n
"
,
token
);
return
;
}
pciBusTceTableParms
.
busNumber
=
token
;
struct
iSeries_Device_Node
*
DevNode
=
(
struct
iSeries_Device_Node
*
)
token
;
pciBusTceTableParms
.
virtualBusFlag
=
0
;
getTceTableParmsiSeries
(
DevNode
,
newTceTable
);
parmsPtr
=
virt_to_absolute
(
(
u64
)
&
pciBusTceTableParms
);
/*
/* Look for existing TCE table for this device. */
* Call HV with the architected data structure to get TCE table
DevNode
->
DevTceTable
=
findHwTceTable
(
newTceTable
);
* info. Put the returned data into the Linux representation
if
(
DevNode
->
DevTceTable
==
NULL
)
{
* of the TCE table data.
DevNode
->
DevTceTable
=
build_tce_table
(
newTceTable
);
*/
HvCallXm_getTceTableParms
(
parmsPtr
);
printk
(
"PCI: getTceTableParms: Bus: 0x%lx Size: 0x%lx, Start: 0x%lx, Index: 0x%lx
\n
"
,
pciBusTceTableParms
.
busNumber
,
pciBusTceTableParms
.
size
,
pciBusTceTableParms
.
startOffset
,
pciBusTceTableParms
.
index
);
/* Determine if the table identified by the index and startOffset */
/* returned by the hypervisor for this bus has already been created. */
/* If so, set the tceTable entry to point to the linux shared tceTable.*/
int
BusIndex
;
for
(
BusIndex
=
0
;
BusIndex
<
255
;
++
BusIndex
)
{
if
(
tceTables
[
BusIndex
]
!=
NULL
)
{
struct
TceTable
*
CmprTceTable
=
tceTables
[
BusIndex
];
if
(
(
CmprTceTable
->
index
==
pciBusTceTableParms
.
index
)
&&
(
CmprTceTable
->
startOffset
==
pciBusTceTableParms
.
startOffset
)
)
{
tceTables
[
token
]
=
CmprTceTable
;
printk
(
"PCI: Bus %lu Shares a TCE table with bus %d
\n
"
,
token
,
BusIndex
);
break
;
}
}
}
/* No shared table, build a new table for this bus. */
if
(
tceTables
[
token
]
==
NULL
)
{
newTceTable
->
size
=
pciBusTceTableParms
.
size
;
newTceTable
->
busNumber
=
pciBusTceTableParms
.
busNumber
;
newTceTable
->
startOffset
=
pciBusTceTableParms
.
startOffset
;
newTceTable
->
index
=
pciBusTceTableParms
.
index
;
builtTceTable
=
build_tce_table
(
newTceTable
);
builtTceTable
->
tceType
=
TCE_PCI
;
tceTables
[
token
]
=
builtTceTable
;
}
}
else
{
else
{
/* We're using
the shared table, not this new one.
*/
/* We're using
a shared table, free this new one.
*/
kfree
(
newTceTable
);
kfree
(
newTceTable
);
}
}
printk
(
"Pci Device 0x%p TceTable: %p
\n
"
,
DevNode
,
DevNode
->
DevTceTable
);
printk
(
"PCI: Pci bus %lu TceTable: %p
\n
"
,
token
,
tceTables
[
token
]);
return
;
return
;
}
else
{
}
/* pSeries Leg */
else
{
struct
device_node
*
dn
;
struct
device_node
*
dn
;
struct
pci_controller
*
phb
;
struct
pci_controller
*
phb
;
...
@@ -864,15 +804,85 @@ void create_pci_bus_tce_table( unsigned long token ) {
...
@@ -864,15 +804,85 @@ void create_pci_bus_tce_table( unsigned long token ) {
getTceTableParmsPSeries
(
phb
,
dn
,
newTceTable
);
getTceTableParmsPSeries
(
phb
,
dn
,
newTceTable
);
else
else
getTceTableParmsPSeriesLP
(
phb
,
dn
,
newTceTable
);
getTceTableParmsPSeriesLP
(
phb
,
dn
,
newTceTable
);
builtTceTable
=
build_tce_table
(
newTceTable
);
dn
->
tce_table
=
builtTceTable
;
dn
->
tce_table
=
build_tce_table
(
newTceTable
)
;
}
}
}
if
(
builtTceTable
==
NULL
)
{
kfree
(
newTceTable
);
/***********************************************************************/
PPCDBG
(
PPCDBG_TCE
,
"PCI Bus TCE table failed.
\n
"
);
/* This function compares the known Tce tables to find a TceTable that */
return
;
/* has already been built for hardware TCEs. */
/* Search the complete(all devices) for a TCE table assigned. If the */
/* startOffset, index, and size match, then the TCE for this device has*/
/* already been built and it should be shared with this device */
/***********************************************************************/
static
struct
TceTable
*
findHwTceTable
(
struct
TceTable
*
newTceTable
)
{
struct
list_head
*
Device_Node_Ptr
=
iSeries_Global_Device_List
.
next
;
/* Cache the compare values. */
u64
startOffset
=
newTceTable
->
startOffset
;
u64
index
=
newTceTable
->
index
;
u64
size
=
newTceTable
->
size
;
while
(
Device_Node_Ptr
!=
&
iSeries_Global_Device_List
)
{
struct
iSeries_Device_Node
*
CmprNode
=
(
struct
iSeries_Device_Node
*
)
Device_Node_Ptr
;
if
(
CmprNode
->
DevTceTable
!=
NULL
&&
CmprNode
->
DevTceTable
->
tceType
==
TCE_PCI
)
{
if
(
CmprNode
->
DevTceTable
->
startOffset
==
startOffset
&&
CmprNode
->
DevTceTable
->
index
==
index
&&
CmprNode
->
DevTceTable
->
size
==
size
)
{
printk
(
"PCI TCE table matches 0x%p
\n
"
,
CmprNode
->
DevTceTable
);
return
CmprNode
->
DevTceTable
;
}
}
/* Get next Device Node in List */
Device_Node_Ptr
=
Device_Node_Ptr
->
next
;
}
}
return
NULL
;
}
/***********************************************************************/
/* Call Hv with the architected data structure to get TCE table info. */
/* info. Put the returned data into the Linux representation of the */
/* TCE table data. */
/* The Hardware Tce table comes in three flavors. */
/* 1. TCE table shared between Buses. */
/* 2. TCE table per Bus. */
/* 3. TCE Table per IOA. */
/***********************************************************************/
static
void
getTceTableParmsiSeries
(
struct
iSeries_Device_Node
*
DevNode
,
struct
TceTable
*
newTceTable
)
{
struct
TceTableManagerCB
*
pciBusTceTableParms
=
(
struct
TceTableManagerCB
*
)
kmalloc
(
sizeof
(
struct
TceTableManagerCB
),
GFP_KERNEL
);
if
(
pciBusTceTableParms
==
NULL
)
panic
(
"PCI_DMA: TCE Table Allocation failed."
);
memset
(
(
void
*
)
pciBusTceTableParms
,
0
,
sizeof
(
struct
TceTableManagerCB
)
);
pciBusTceTableParms
->
busNumber
=
ISERIES_BUS
(
DevNode
);
pciBusTceTableParms
->
logicalSlot
=
DevNode
->
LogicalSlot
;
pciBusTceTableParms
->
virtualBusFlag
=
0
;
HvCallXm_getTceTableParms
(
REALADDR
(
pciBusTceTableParms
)
);
/* PciTceTableParms Bus:0x18 Slot:0x04 Start:0x000000 Offset:0x04c000 Size:0x0020 */
printk
(
"PciTceTableParms Bus:0x%02lx Slot:0x%02x Start:0x%06lx Offset:0x%06lx Size:0x%04lx
\n
"
,
pciBusTceTableParms
->
busNumber
,
pciBusTceTableParms
->
logicalSlot
,
pciBusTceTableParms
->
start
,
pciBusTceTableParms
->
startOffset
,
pciBusTceTableParms
->
size
);
if
(
pciBusTceTableParms
->
size
==
0
)
{
printk
(
"PCI_DMA: Possible Structure mismatch, 0x%p
\n
"
,
pciBusTceTableParms
);
panic
(
"PCI_DMA: pciBusTceTableParms->size is zero, halt here!"
);
}
newTceTable
->
size
=
pciBusTceTableParms
->
size
;
newTceTable
->
busNumber
=
pciBusTceTableParms
->
busNumber
;
newTceTable
->
startOffset
=
pciBusTceTableParms
->
startOffset
;
newTceTable
->
index
=
pciBusTceTableParms
->
index
;
newTceTable
->
tceType
=
TCE_PCI
;
kfree
(
pciBusTceTableParms
);
}
}
static
void
getTceTableParmsPSeries
(
struct
pci_controller
*
phb
,
static
void
getTceTableParmsPSeries
(
struct
pci_controller
*
phb
,
...
@@ -927,8 +937,7 @@ static void getTceTableParmsPSeries(struct pci_controller *phb,
...
@@ -927,8 +937,7 @@ static void getTceTableParmsPSeries(struct pci_controller *phb,
/* Test if we are going over 2GB of DMA space. */
/* Test if we are going over 2GB of DMA space. */
if
(
phb
->
dma_window_base_cur
>
(
1
<<
19
))
{
if
(
phb
->
dma_window_base_cur
>
(
1
<<
19
))
{
udbg_printf
(
"Unexpected number of IOAs under this PHB"
);
panic
(
"PCI_DMA: Unexpected number of IOAs under this PHB.
\n
"
);
panic
(
"Unexpected number of IOAs under this PHB"
);
}
}
newTceTable
->
base
=
of_tce_table
[
i
].
base
;
newTceTable
->
base
=
of_tce_table
[
i
].
base
;
...
@@ -967,7 +976,7 @@ static void getTceTableParmsPSeriesLP(struct pci_controller *phb,
...
@@ -967,7 +976,7 @@ static void getTceTableParmsPSeriesLP(struct pci_controller *phb,
struct
TceTable
*
newTceTable
)
{
struct
TceTable
*
newTceTable
)
{
u32
*
dma_window
=
(
u32
*
)
get_property
(
dn
,
"ibm,dma-window"
,
0
);
u32
*
dma_window
=
(
u32
*
)
get_property
(
dn
,
"ibm,dma-window"
,
0
);
if
(
!
dma_window
)
{
if
(
!
dma_window
)
{
panic
(
"
getTceTableParmsPSeriesLP:
device %s has no ibm,dma-window property!
\n
"
,
dn
->
full_name
);
panic
(
"
PCI_DMA: getTceTableParmsPSeriesLP:
device %s has no ibm,dma-window property!
\n
"
,
dn
->
full_name
);
}
}
newTceTable
->
busNumber
=
dn
->
busno
;
newTceTable
->
busNumber
=
dn
->
busno
;
...
@@ -1003,6 +1012,13 @@ void *pci_alloc_consistent(struct pci_dev *hwdev, size_t size,
...
@@ -1003,6 +1012,13 @@ void *pci_alloc_consistent(struct pci_dev *hwdev, size_t size,
order
=
get_order
(
size
);
order
=
get_order
(
size
);
nPages
=
1
<<
order
;
nPages
=
1
<<
order
;
/* Client asked for way to much space. This is checked later anyway */
/* It is easier to debug here for the drivers than in the tce tables.*/
if
(
order
>=
NUM_TCE_LEVELS
)
{
printk
(
"PCI_DMA: pci_alloc_consistent size to large: 0x%lx
\n
"
,
size
);
return
(
void
*
)
NO_TCE
;
}
tbl
=
get_tce_table
(
hwdev
);
tbl
=
get_tce_table
(
hwdev
);
if
(
tbl
)
{
if
(
tbl
)
{
...
@@ -1045,14 +1061,17 @@ void pci_free_consistent(struct pci_dev *hwdev, size_t size,
...
@@ -1045,14 +1061,17 @@ void pci_free_consistent(struct pci_dev *hwdev, size_t size,
order
=
get_order
(
size
);
order
=
get_order
(
size
);
nPages
=
1
<<
order
;
nPages
=
1
<<
order
;
if
(
order
>
10
)
/* Client asked for way to much space. This is checked later anyway */
PPCDBG
(
PPCDBG_TCE
,
"pci_free_consistent: order=%d, size=%d, nPages=%d, dma_handle=%016lx, vaddr=%016lx
\n
"
,
/* It is easier to debug here for the drivers than in the tce tables.*/
order
,
size
,
nPages
,
(
unsigned
long
)
dma_handle
,
(
unsigned
long
)
vaddr
);
if
(
order
>=
NUM_TCE_LEVELS
)
{
printk
(
"PCI_DMA: pci_free_consistent size to large: 0x%lx
\n
"
,
size
);
return
;
}
tbl
=
get_tce_table
(
hwdev
);
tbl
=
get_tce_table
(
hwdev
);
if
(
tbl
)
{
if
(
tbl
)
{
ppc_md
.
tce_free
(
tbl
,
dma_handle
,
order
,
nPages
);
tce_free
(
tbl
,
dma_handle
,
order
,
nPages
);
free_pages
(
(
unsigned
long
)
vaddr
,
order
);
free_pages
(
(
unsigned
long
)
vaddr
,
order
);
}
}
}
}
...
@@ -1081,6 +1100,13 @@ dma_addr_t pci_map_single(struct pci_dev *hwdev, void *vaddr,
...
@@ -1081,6 +1100,13 @@ dma_addr_t pci_map_single(struct pci_dev *hwdev, void *vaddr,
order
=
get_order
(
nPages
&
PAGE_MASK
);
order
=
get_order
(
nPages
&
PAGE_MASK
);
nPages
>>=
PAGE_SHIFT
;
nPages
>>=
PAGE_SHIFT
;
/* Client asked for way to much space. This is checked later anyway */
/* It is easier to debug here for the drivers than in the tce tables.*/
if
(
order
>=
NUM_TCE_LEVELS
)
{
printk
(
"PCI_DMA: pci_map_single size to large: 0x%lx
\n
"
,
size
);
return
NO_TCE
;
}
tbl
=
get_tce_table
(
hwdev
);
tbl
=
get_tce_table
(
hwdev
);
if
(
tbl
)
{
if
(
tbl
)
{
...
@@ -1105,14 +1131,17 @@ void pci_unmap_single( struct pci_dev *hwdev, dma_addr_t dma_handle, size_t size
...
@@ -1105,14 +1131,17 @@ void pci_unmap_single( struct pci_dev *hwdev, dma_addr_t dma_handle, size_t size
order
=
get_order
(
nPages
&
PAGE_MASK
);
order
=
get_order
(
nPages
&
PAGE_MASK
);
nPages
>>=
PAGE_SHIFT
;
nPages
>>=
PAGE_SHIFT
;
if
(
order
>
10
)
/* Client asked for way to much space. This is checked later anyway */
PPCDBG
(
PPCDBG_TCE
,
"pci_unmap_single: order=%d, size=%d, nPages=%d, dma_handle=%016lx
\n
"
,
/* It is easier to debug here for the drivers than in the tce tables.*/
order
,
size
,
nPages
,
(
unsigned
long
)
dma_handle
);
if
(
order
>=
NUM_TCE_LEVELS
)
{
printk
(
"PCI_DMA: pci_unmap_single size to large: 0x%lx
\n
"
,
size
);
return
;
}
tbl
=
get_tce_table
(
hwdev
);
tbl
=
get_tce_table
(
hwdev
);
if
(
tbl
)
if
(
tbl
)
ppc_md
.
tce_free
(
tbl
,
dma_handle
,
order
,
nPages
);
tce_free
(
tbl
,
dma_handle
,
order
,
nPages
);
}
}
...
@@ -1261,6 +1290,13 @@ static dma_addr_t create_tces_sg( struct TceTable *tbl, struct scatterlist *sg,
...
@@ -1261,6 +1290,13 @@ static dma_addr_t create_tces_sg( struct TceTable *tbl, struct scatterlist *sg,
dmaAddr = NO_TCE;
dmaAddr = NO_TCE;
order = get_order( numTces << PAGE_SHIFT );
order = get_order( numTces << PAGE_SHIFT );
/* Client asked for way to much space. This is checked later anyway */
/* It is easier to debug here for the drivers than in the tce tables.*/
if(order >= NUM_TCE_LEVELS) {
printk("PCI_DMA: create_tces_sg size to large: 0x%x \n",(numTces << PAGE_SHIFT));
return NO_TCE;
}
/* allocate a block of tces */
/* allocate a block of tces */
tcenum = alloc_tce_range( tbl, order );
tcenum = alloc_tce_range( tbl, order );
if ( tcenum != -1 ) {
if ( tcenum != -1 ) {
...
@@ -1291,10 +1327,16 @@ static dma_addr_t create_tces_sg( struct TceTable *tbl, struct scatterlist *sg,
...
@@ -1291,10 +1327,16 @@ static dma_addr_t create_tces_sg( struct TceTable *tbl, struct scatterlist *sg,
++tcenum;
++tcenum;
uaddr += PAGE_SIZE;
uaddr += PAGE_SIZE;
}
}
prevEndPage = endPage;
prevEndPage = endPage;
sg++;
sg++;
}
}
/* Make sure the update is visible to hardware.
sync required to synchronize the update to
the TCE table with the MMIO that will send
the bus address to the IOA */
__asm__ __volatile__ ("sync" : : : "memory");
if ((tcenum - starttcenum) != numTces)
if ((tcenum - starttcenum) != numTces)
PPCDBG(PPCDBG_TCE, "create_tces_sg: numTces %d, tces used %d\n",
PPCDBG(PPCDBG_TCE, "create_tces_sg: numTces %d, tces used %d\n",
numTces, (unsigned)(tcenum - starttcenum));
numTces, (unsigned)(tcenum - starttcenum));
...
@@ -1364,14 +1406,17 @@ void pci_unmap_sg( struct pci_dev *hwdev, struct scatterlist *sg, int nelms, int
...
@@ -1364,14 +1406,17 @@ void pci_unmap_sg( struct pci_dev *hwdev, struct scatterlist *sg, int nelms, int
numTces = ((dma_end_page - dma_start_page ) >> PAGE_SHIFT) + 1;
numTces = ((dma_end_page - dma_start_page ) >> PAGE_SHIFT) + 1;
order = get_order( numTces << PAGE_SHIFT );
order = get_order( numTces << PAGE_SHIFT );
if ( order > 10 )
/* Client asked for way to much space. This is checked later anyway */
PPCDBG(PPCDBG_TCE, "pci_unmap_sg: order=%d, numTces=%d, nelms=%d, dma_start_page=%016lx, dma_end_page=%016lx\n",
/* It is easier to debug here for the drivers than in the tce tables.*/
order, numTces, nelms, (unsigned long)dma_start_page, (unsigned long)dma_end_page );
if(order >= NUM_TCE_LEVELS) {
printk("PCI_DMA: pci_unmap_sg size to large: 0x%x \n",(numTces << PAGE_SHIFT));
return;
}
tbl = get_tce_table(hwdev);
tbl = get_tce_table(hwdev);
if ( tbl )
if ( tbl )
ppc_md.
tce_free( tbl, dma_start_page, order, numTces );
tce_free( tbl, dma_start_page, order, numTces );
}
}
#else
#else
...
@@ -1385,7 +1430,8 @@ int pci_map_sg(struct pci_dev *pdev, struct scatterlist *sglist, int nelems,
...
@@ -1385,7 +1430,8 @@ int pci_map_sg(struct pci_dev *pdev, struct scatterlist *sglist, int nelems,
sglist
->
offset
);
sglist
->
offset
);
sglist
->
dma_address
=
pci_map_single
(
pdev
,
vaddr
,
sglist
->
dma_address
=
pci_map_single
(
pdev
,
vaddr
,
sglist
->
length
,
direction
);
sglist
->
length
,
direction
);
sglist
->
dma_length
=
sglist
->
length
;
sglist
->
dma_length
=
sglist
->
length
;
sglist
++
;
sglist
++
;
}
}
...
@@ -1471,11 +1517,11 @@ unsigned long phb_tce_table_init(struct pci_controller *phb) {
...
@@ -1471,11 +1517,11 @@ unsigned long phb_tce_table_init(struct pci_controller *phb) {
void
tce_init_pSeries
(
void
)
void
tce_init_pSeries
(
void
)
{
{
ppc_md
.
tce_build
=
tce_build_pSeries
;
ppc_md
.
tce_build
=
tce_build_pSeries
;
ppc_md
.
tce_free
=
tce_fre
e_pSeries
;
ppc_md
.
tce_free
_one
=
tce_free_on
e_pSeries
;
}
}
void
tce_init_iSeries
(
void
)
void
tce_init_iSeries
(
void
)
{
{
ppc_md
.
tce_build
=
tce_build_iSeries
;
ppc_md
.
tce_build
=
tce_build_iSeries
;
ppc_md
.
tce_free
=
tce_fre
e_iSeries
;
ppc_md
.
tce_free
_one
=
tce_free_on
e_iSeries
;
}
}
include/asm-ppc64/machdep.h
View file @
34f16cd8
...
@@ -55,10 +55,8 @@ struct machdep_calls {
...
@@ -55,10 +55,8 @@ struct machdep_calls {
long
tcenum
,
long
tcenum
,
unsigned
long
uaddr
,
unsigned
long
uaddr
,
int
direction
);
int
direction
);
void
(
*
tce_free
)(
struct
TceTable
*
tbl
,
void
(
*
tce_free_one
)(
struct
TceTable
*
tbl
,
dma_addr_t
dma_addr
,
long
tcenum
);
unsigned
order
,
unsigned
numPages
);
void
(
*
smp_message_pass
)(
int
target
,
void
(
*
smp_message_pass
)(
int
target
,
int
msg
,
int
msg
,
...
...
include/asm-ppc64/pci_dma.h
View file @
34f16cd8
...
@@ -85,12 +85,14 @@ struct TceTableManagerCB {
...
@@ -85,12 +85,14 @@ struct TceTableManagerCB {
u64
index
;
/* Index of this tce table (token?) */
u64
index
;
/* Index of this tce table (token?) */
u16
maxTceTableIndex
;
/* Max num of tables for partition */
u16
maxTceTableIndex
;
/* Max num of tables for partition */
u8
virtualBusFlag
;
/* Flag to indicate virtual bus */
u8
virtualBusFlag
;
/* Flag to indicate virtual bus */
u8
rsvd
[
5
];
u8
logicalSlot
;
/* IOA Tce Slot Index */
u8
rsvd
[
4
];
};
};
extern
struct
TceTable
virtBusTceTable
;
/* Tce table for virtual bus */
extern
struct
TceTable
virtBusTceTable
;
/* Tce table for virtual bus */
extern
void
create_tce_tables
(
void
);
extern
void
create_tce_tables
(
void
);
extern
void
create_pci_bus_tce_table
(
unsigned
long
);
void
tce_init_pSeries
(
void
);
void
tce_init_pSeries
(
void
);
void
tce_init_iSeries
(
void
);
void
tce_init_iSeries
(
void
);
...
...
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