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
d0ac249b
Commit
d0ac249b
authored
Jun 05, 2002
by
Rob Radez
Committed by
David S. Miller
Jun 05, 2002
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Sparc32 code cleanups from 2.4.x.
parent
a9907091
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
115 additions
and
570 deletions
+115
-570
arch/sparc/kernel/devices.c
arch/sparc/kernel/devices.c
+28
-29
arch/sparc/mm/sun4c.c
arch/sparc/mm/sun4c.c
+71
-460
include/asm-sparc/btfixup.h
include/asm-sparc/btfixup.h
+16
-27
include/asm-sparc/ultra.h
include/asm-sparc/ultra.h
+0
-52
include/asm-sparc/vac-ops.h
include/asm-sparc/vac-ops.h
+0
-2
No files found.
arch/sparc/kernel/devices.c
View file @
d0ac249b
...
@@ -22,8 +22,8 @@ extern void cpu_probe(void);
...
@@ -22,8 +22,8 @@ extern void cpu_probe(void);
extern
void
clock_stop_probe
(
void
);
/* tadpole.c */
extern
void
clock_stop_probe
(
void
);
/* tadpole.c */
extern
void
sun4c_probe_memerr_reg
(
void
);
extern
void
sun4c_probe_memerr_reg
(
void
);
unsigned
long
__init
void
__init
device_scan
(
unsigned
long
mem_start
)
device_scan
(
void
)
{
{
char
node_str
[
128
];
char
node_str
[
128
];
int
thismid
;
int
thismid
;
...
@@ -37,46 +37,45 @@ device_scan(unsigned long mem_start)
...
@@ -37,46 +37,45 @@ device_scan(unsigned long mem_start)
int
scan
;
int
scan
;
scan
=
prom_getchild
(
prom_root_node
);
scan
=
prom_getchild
(
prom_root_node
);
/* One can look it up in PROM instead */
/* One can look it up in PROM instead */
/* prom_printf("root child is %08lx\n", (unsigned long) scan); */
while
((
scan
=
prom_getsibling
(
scan
))
!=
0
)
{
while
((
scan
=
prom_getsibling
(
scan
))
!=
0
)
{
prom_getstring
(
scan
,
"device_type"
,
prom_getstring
(
scan
,
"device_type"
,
node_str
,
sizeof
(
node_str
));
node_str
,
sizeof
(
node_str
));
if
(
strcmp
(
node_str
,
"cpu"
)
==
0
)
{
if
(
strcmp
(
node_str
,
"cpu"
)
==
0
)
{
linux_cpus
[
linux_num_cpus
].
prom_node
=
scan
;
linux_cpus
[
linux_num_cpus
].
prom_node
=
scan
;
prom_getproperty
(
scan
,
"mid"
,
(
char
*
)
&
thismid
,
sizeof
(
thismid
));
prom_getproperty
(
scan
,
"mid"
,
(
char
*
)
&
thismid
,
sizeof
(
thismid
));
linux_cpus
[
linux_num_cpus
].
mid
=
thismid
;
linux_cpus
[
linux_num_cpus
].
mid
=
thismid
;
/* prom_printf("Found CPU %d <node=%08lx,mid=%d>\n", linux_num_cpus, (unsigned long) scan, thismid); */
printk
(
"Found CPU %d <node=%08lx,mid=%d>
\n
"
,
printk
(
"Found CPU %d <node=%08lx,mid=%d>
\n
"
,
linux_num_cpus
,
(
unsigned
long
)
scan
,
thismid
);
linux_num_cpus
,
(
unsigned
long
)
scan
,
thismid
);
linux_num_cpus
++
;
linux_num_cpus
++
;
}
}
}
}
if
(
linux_num_cpus
==
0
)
{
if
(
linux_num_cpus
==
0
&&
sparc_cpu_model
==
sun4d
)
{
if
(
sparc_cpu_model
==
sun4d
)
{
scan
=
prom_getchild
(
prom_root_node
);
scan
=
prom_getchild
(
prom_root_node
);
for
(
scan
=
prom_searchsiblings
(
scan
,
"cpu-unit"
);
scan
;
for
(
scan
=
prom_searchsiblings
(
scan
,
"cpu-unit"
);
scan
;
scan
=
prom_searchsiblings
(
prom_getsibling
(
scan
),
"cpu-unit"
))
{
scan
=
prom_searchsiblings
(
prom_getsibling
(
scan
),
"cpu-unit"
))
{
int
node
=
prom_getchild
(
scan
);
int
node
=
prom_getchild
(
scan
);
prom_getstring
(
node
,
"device_type"
,
node_str
,
sizeof
(
node_str
));
prom_getstring
(
node
,
"device_type"
,
if
(
strcmp
(
node_str
,
"cpu"
)
==
0
)
{
node_str
,
sizeof
(
node_str
));
prom_getproperty
(
node
,
"cpu-id"
,
(
char
*
)
&
thismid
,
sizeof
(
thismid
));
if
(
strcmp
(
node_str
,
"cpu"
)
==
0
)
{
linux_cpus
[
linux_num_cpus
].
prom_node
=
node
;
prom_getproperty
(
node
,
"cpu-id"
,
linux_cpus
[
linux_num_cpus
].
mid
=
thismid
;
(
char
*
)
&
thismid
,
sizeof
(
thismid
));
/* prom_printf("Found CPU %d <node=%08lx,mid=%d>\n",
linux_cpus
[
linux_num_cpus
].
prom_node
=
node
;
linux_num_cpus, (unsigned long) node, thismid); */
linux_cpus
[
linux_num_cpus
].
mid
=
thismid
;
printk
(
"Found CPU %d <node=%08lx,mid=%d>
\n
"
,
printk
(
"Found CPU %d <node=%08lx,mid=%d>
\n
"
,
linux_num_cpus
,
(
unsigned
long
)
node
,
thismid
);
linux_num_cpus
,
(
unsigned
long
)
node
,
thismid
);
linux_num_cpus
++
;
linux_num_cpus
++
;
}
}
}
}
}
}
}
if
(
linux_num_cpus
==
0
)
{
if
(
linux_num_cpus
==
0
)
{
printk
(
"No CPU nodes found, cannot continue.
\n
"
);
printk
(
"No CPU nodes found, cannot continue.
\n
"
);
/* Probably a sun4e, Sun is trying to trick us ;-) */
/* Probably a sun4e, Sun is trying to trick us ;-) */
halt
();
halt
();
}
}
printk
(
"Found %d CPU prom device tree node(s).
\n
"
,
linux_num_cpus
);
printk
(
"Found %d CPU prom device tree node(s).
\n
"
,
linux_num_cpus
);
}
;
}
cpu_probe
();
cpu_probe
();
#ifdef CONFIG_SUN_AUXIO
#ifdef CONFIG_SUN_AUXIO
...
@@ -92,5 +91,5 @@ device_scan(unsigned long mem_start)
...
@@ -92,5 +91,5 @@ device_scan(unsigned long mem_start)
if
(
ARCH_SUN4C_SUN4
)
if
(
ARCH_SUN4C_SUN4
)
sun4c_probe_memerr_reg
();
sun4c_probe_memerr_reg
();
return
mem_start
;
return
;
}
}
arch/sparc/mm/sun4c.c
View file @
d0ac249b
...
@@ -35,6 +35,7 @@
...
@@ -35,6 +35,7 @@
#include <asm/mmu_context.h>
#include <asm/mmu_context.h>
#include <asm/sun4paddr.h>
#include <asm/sun4paddr.h>
#include <asm/highmem.h>
#include <asm/highmem.h>
#include <asm/btfixup.h>
/* Because of our dynamic kernel TLB miss strategy, and how
/* Because of our dynamic kernel TLB miss strategy, and how
* our DVMA mapping allocation works, you _MUST_:
* our DVMA mapping allocation works, you _MUST_:
...
@@ -63,25 +64,17 @@ extern unsigned long page_kernel;
...
@@ -63,25 +64,17 @@ extern unsigned long page_kernel;
#define SUN4C_KERNEL_BUCKETS 32
#define SUN4C_KERNEL_BUCKETS 32
#ifndef MAX
#define MAX(a,b) ((a)<(b)?(b):(a))
#endif
#ifndef MIN
#define MIN(a,b) ((a)<(b)?(a):(b))
#endif
/* Flushing the cache. */
/* Flushing the cache. */
struct
sun4c_vac_props
sun4c_vacinfo
;
struct
sun4c_vac_props
sun4c_vacinfo
;
unsigned
long
sun4c_kernel_faults
;
unsigned
long
sun4c_kernel_faults
;
/* Invalidate every sun4c cache line tag. */
/* Invalidate every sun4c cache line tag. */
void
sun4c_flush_all
(
void
)
static
void
__init
sun4c_flush_all
(
void
)
{
{
unsigned
long
begin
,
end
;
unsigned
long
begin
,
end
;
if
(
sun4c_vacinfo
.
on
)
if
(
sun4c_vacinfo
.
on
)
panic
(
"SUN4C: AIEEE, trying to invalidate vac while"
panic
(
"SUN4C: AIEEE, trying to invalidate vac while it is on."
);
" it is on."
);
/* Clear 'valid' bit in all cache line tags */
/* Clear 'valid' bit in all cache line tags */
begin
=
AC_CACHETAGS
;
begin
=
AC_CACHETAGS
;
...
@@ -93,7 +86,7 @@ void sun4c_flush_all(void)
...
@@ -93,7 +86,7 @@ void sun4c_flush_all(void)
}
}
}
}
static
__inline__
void
sun4c_flush_context_hw
(
void
)
static
void
sun4c_flush_context_hw
(
void
)
{
{
unsigned
long
end
=
SUN4C_VAC_SIZE
;
unsigned
long
end
=
SUN4C_VAC_SIZE
;
...
@@ -122,8 +115,17 @@ static void sun4c_flush_segment_hw(unsigned long addr)
...
@@ -122,8 +115,17 @@ static void sun4c_flush_segment_hw(unsigned long addr)
}
}
}
}
/* File local boot time fixups. */
BTFIXUPDEF_CALL
(
void
,
sun4c_flush_page
,
unsigned
long
)
BTFIXUPDEF_CALL
(
void
,
sun4c_flush_segment
,
unsigned
long
)
BTFIXUPDEF_CALL
(
void
,
sun4c_flush_context
,
void
)
#define sun4c_flush_page(addr) BTFIXUP_CALL(sun4c_flush_page)(addr)
#define sun4c_flush_segment(addr) BTFIXUP_CALL(sun4c_flush_segment)(addr)
#define sun4c_flush_context() BTFIXUP_CALL(sun4c_flush_context)()
/* Must be called minimally with interrupts disabled. */
/* Must be called minimally with interrupts disabled. */
static
__inline__
void
sun4c_flush_page_hw
(
unsigned
long
addr
)
static
void
sun4c_flush_page_hw
(
unsigned
long
addr
)
{
{
addr
&=
PAGE_MASK
;
addr
&=
PAGE_MASK
;
if
((
int
)
sun4c_get_pte
(
addr
)
<
0
)
if
((
int
)
sun4c_get_pte
(
addr
)
<
0
)
...
@@ -195,48 +197,6 @@ static void sun4c_flush_segment_sw(unsigned long addr)
...
@@ -195,48 +197,6 @@ static void sun4c_flush_segment_sw(unsigned long addr)
}
}
}
}
/* Bolix one page from the virtual cache. */
static
void
sun4c_flush_page
(
unsigned
long
addr
)
{
addr
&=
PAGE_MASK
;
if
((
sun4c_get_pte
(
addr
)
&
(
_SUN4C_PAGE_NOCACHE
|
_SUN4C_PAGE_VALID
))
!=
_SUN4C_PAGE_VALID
)
return
;
if
(
sun4c_vacinfo
.
do_hwflushes
)
{
__asm__
__volatile__
(
"sta %%g0, [%0] %1;nop;nop;nop;
\n\t
"
:
:
"r"
(
addr
),
"i"
(
ASI_HWFLUSHPAGE
));
}
else
{
unsigned
long
left
=
PAGE_SIZE
;
unsigned
long
lsize
=
sun4c_vacinfo
.
linesize
;
__asm__
__volatile__
(
"add %2, %2, %%g1
\n\t
"
"add %2, %%g1, %%g2
\n\t
"
"add %2, %%g2, %%g3
\n\t
"
"add %2, %%g3, %%g4
\n\t
"
"add %2, %%g4, %%g5
\n\t
"
"add %2, %%g5, %%o4
\n\t
"
"add %2, %%o4, %%o5
\n
"
"1:
\n\t
"
"subcc %1, %%o5, %1
\n\t
"
"sta %%g0, [%0] %6
\n\t
"
"sta %%g0, [%0 + %2] %6
\n\t
"
"sta %%g0, [%0 + %%g1] %6
\n\t
"
"sta %%g0, [%0 + %%g2] %6
\n\t
"
"sta %%g0, [%0 + %%g3] %6
\n\t
"
"sta %%g0, [%0 + %%g4] %6
\n\t
"
"sta %%g0, [%0 + %%g5] %6
\n\t
"
"sta %%g0, [%0 + %%o4] %6
\n\t
"
"bg 1b
\n\t
"
" add %0, %%o5, %0
\n\t
"
:
"=&r"
(
addr
),
"=&r"
(
left
),
"=&r"
(
lsize
)
:
"0"
(
addr
),
"1"
(
left
),
"2"
(
lsize
),
"i"
(
ASI_FLUSHPG
)
:
"g1"
,
"g2"
,
"g3"
,
"g4"
,
"g5"
,
"o4"
,
"o5"
,
"cc"
);
}
}
/* Don't inline the software version as it eats too many cache lines if expanded. */
/* Don't inline the software version as it eats too many cache lines if expanded. */
static
void
sun4c_flush_page_sw
(
unsigned
long
addr
)
static
void
sun4c_flush_page_sw
(
unsigned
long
addr
)
{
{
...
@@ -387,7 +347,8 @@ void __init sun4c_probe_vac(void)
...
@@ -387,7 +347,8 @@ void __init sun4c_probe_vac(void)
prom_getintdefault
(
prom_root_node
,
"vac_hwflush"
,
0
);
prom_getintdefault
(
prom_root_node
,
"vac_hwflush"
,
0
);
if
(
sun4c_vacinfo
.
num_bytes
!=
65536
)
{
if
(
sun4c_vacinfo
.
num_bytes
!=
65536
)
{
prom_printf
(
"WEIRD Sun4C VAC cache size, tell davem"
);
prom_printf
(
"WEIRD Sun4C VAC cache size, "
"tell sparclinux@vger.kernel.org"
);
prom_halt
();
prom_halt
();
}
}
}
}
...
@@ -427,7 +388,7 @@ extern unsigned long vac_hwflush_patch2, vac_hwflush_patch2_on;
...
@@ -427,7 +388,7 @@ extern unsigned long vac_hwflush_patch2, vac_hwflush_patch2_on;
*daddr = *iaddr; \
*daddr = *iaddr; \
} while (0)
} while (0)
static
void
patch_kernel_fault_handler
(
void
)
static
void
__init
patch_kernel_fault_handler
(
void
)
{
{
unsigned
long
*
iaddr
,
*
daddr
;
unsigned
long
*
iaddr
,
*
daddr
;
...
@@ -459,10 +420,6 @@ static void patch_kernel_fault_handler(void)
...
@@ -459,10 +420,6 @@ static void patch_kernel_fault_handler(void)
case
16
:
case
16
:
PATCH_INSN
(
num_context_patch1_16
,
PATCH_INSN
(
num_context_patch1_16
,
num_context_patch1
);
num_context_patch1
);
#if 0
PATCH_INSN(num_context_patch2_16,
num_context_patch2);
#endif
break
;
break
;
default:
default:
prom_printf
(
"Unhandled number of contexts: %d
\n
"
,
prom_printf
(
"Unhandled number of contexts: %d
\n
"
,
...
@@ -867,7 +824,7 @@ static void sun4c_kernel_map(struct sun4c_mmu_entry *kentry)
...
@@ -867,7 +824,7 @@ static void sun4c_kernel_map(struct sun4c_mmu_entry *kentry)
#define sun4c_user_unmap(__entry) \
#define sun4c_user_unmap(__entry) \
sun4c_put_segmap((__entry)->vaddr, invalid_segment)
sun4c_put_segmap((__entry)->vaddr, invalid_segment)
static
void
sun4c_demap_context
_hw
(
struct
sun4c_mmu_ring
*
crp
,
unsigned
char
ctx
)
static
void
sun4c_demap_context
(
struct
sun4c_mmu_ring
*
crp
,
unsigned
char
ctx
)
{
{
struct
sun4c_mmu_entry
*
head
=
&
crp
->
ringhd
;
struct
sun4c_mmu_entry
*
head
=
&
crp
->
ringhd
;
unsigned
long
flags
;
unsigned
long
flags
;
...
@@ -879,7 +836,7 @@ static void sun4c_demap_context_hw(struct sun4c_mmu_ring *crp, unsigned char ctx
...
@@ -879,7 +836,7 @@ static void sun4c_demap_context_hw(struct sun4c_mmu_ring *crp, unsigned char ctx
flush_user_windows
();
flush_user_windows
();
sun4c_set_context
(
ctx
);
sun4c_set_context
(
ctx
);
sun4c_flush_context
_hw
();
sun4c_flush_context
();
do
{
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
...
@@ -893,34 +850,8 @@ static void sun4c_demap_context_hw(struct sun4c_mmu_ring *crp, unsigned char ctx
...
@@ -893,34 +850,8 @@ static void sun4c_demap_context_hw(struct sun4c_mmu_ring *crp, unsigned char ctx
restore_flags
(
flags
);
restore_flags
(
flags
);
}
}
static
void
sun4c_demap_context_sw
(
struct
sun4c_mmu_ring
*
crp
,
unsigned
char
ctx
)
static
int
sun4c_user_taken_entries
;
/* This is how much we have. */
{
static
int
max_user_taken_entries
;
/* This limits us and prevents deadlock. */
struct
sun4c_mmu_entry
*
head
=
&
crp
->
ringhd
;
unsigned
long
flags
;
save_and_cli
(
flags
);
if
(
head
->
next
!=
head
)
{
struct
sun4c_mmu_entry
*
entry
=
head
->
next
;
int
savectx
=
sun4c_get_context
();
flush_user_windows
();
sun4c_set_context
(
ctx
);
sun4c_flush_context_sw
();
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
sun4c_user_unmap
(
entry
);
free_user_entry
(
ctx
,
entry
);
entry
=
next
;
}
while
(
entry
!=
head
);
sun4c_set_context
(
savectx
);
}
restore_flags
(
flags
);
}
static
int
sun4c_user_taken_entries
=
0
;
/* This is how much we have. */
static
int
max_user_taken_entries
=
0
;
/* This limits us and prevents deadlock. */
static
struct
sun4c_mmu_entry
*
sun4c_kernel_strategy
(
void
)
static
struct
sun4c_mmu_entry
*
sun4c_kernel_strategy
(
void
)
{
{
...
@@ -934,10 +865,7 @@ static struct sun4c_mmu_entry *sun4c_kernel_strategy(void)
...
@@ -934,10 +865,7 @@ static struct sun4c_mmu_entry *sun4c_kernel_strategy(void)
/* Else free one up. */
/* Else free one up. */
this_entry
=
sun4c_kernel_ring
.
ringhd
.
prev
;
this_entry
=
sun4c_kernel_ring
.
ringhd
.
prev
;
if
(
sun4c_vacinfo
.
do_hwflushes
)
sun4c_flush_segment
(
this_entry
->
vaddr
);
sun4c_flush_segment_hw
(
this_entry
->
vaddr
);
else
sun4c_flush_segment_sw
(
this_entry
->
vaddr
);
sun4c_kernel_unmap
(
this_entry
);
sun4c_kernel_unmap
(
this_entry
);
free_kernel_entry
(
this_entry
,
&
sun4c_kernel_ring
);
free_kernel_entry
(
this_entry
,
&
sun4c_kernel_ring
);
this_entry
=
sun4c_kfree_ring
.
ringhd
.
next
;
this_entry
=
sun4c_kfree_ring
.
ringhd
.
next
;
...
@@ -976,10 +904,7 @@ static struct sun4c_mmu_entry *sun4c_user_strategy(void)
...
@@ -976,10 +904,7 @@ static struct sun4c_mmu_entry *sun4c_user_strategy(void)
savectx
=
sun4c_get_context
();
savectx
=
sun4c_get_context
();
flush_user_windows
();
flush_user_windows
();
sun4c_set_context
(
ctx
);
sun4c_set_context
(
ctx
);
if
(
sun4c_vacinfo
.
do_hwflushes
)
sun4c_flush_segment
(
entry
->
vaddr
);
sun4c_flush_segment_hw
(
entry
->
vaddr
);
else
sun4c_flush_segment_sw
(
entry
->
vaddr
);
sun4c_user_unmap
(
entry
);
sun4c_user_unmap
(
entry
);
remove_ring
(
sun4c_context_ring
+
ctx
,
entry
);
remove_ring
(
sun4c_context_ring
+
ctx
,
entry
);
remove_lru
(
entry
);
remove_lru
(
entry
);
...
@@ -1068,10 +993,7 @@ static void free_locked_segment(unsigned long addr)
...
@@ -1068,10 +993,7 @@ static void free_locked_segment(unsigned long addr)
entry
=
&
mmu_entry_pool
[
pseg
];
entry
=
&
mmu_entry_pool
[
pseg
];
flush_user_windows
();
flush_user_windows
();
if
(
sun4c_vacinfo
.
do_hwflushes
)
sun4c_flush_segment
(
addr
);
sun4c_flush_segment_hw
(
addr
);
else
sun4c_flush_segment_sw
(
addr
);
sun4c_kernel_unmap
(
entry
);
sun4c_kernel_unmap
(
entry
);
add_ring
(
&
sun4c_ufree_ring
,
entry
);
add_ring
(
&
sun4c_ufree_ring
,
entry
);
max_user_taken_entries
++
;
max_user_taken_entries
++
;
...
@@ -1126,17 +1048,10 @@ static struct task_struct *sun4c_alloc_task_struct(void)
...
@@ -1126,17 +1048,10 @@ static struct task_struct *sun4c_alloc_task_struct(void)
/* We are changing the virtual color of the page(s)
/* We are changing the virtual color of the page(s)
* so we must flush the cache to guarentee consistancy.
* so we must flush the cache to guarentee consistancy.
*/
*/
if
(
sun4c_vacinfo
.
do_hwflushes
)
{
sun4c_flush_page
(
pages
);
sun4c_flush_page_hw
(
pages
);
#ifndef CONFIG_SUN4
sun4c_flush_page_hw
(
pages
+
PAGE_SIZE
);
#endif
}
else
{
sun4c_flush_page_sw
(
pages
);
#ifndef CONFIG_SUN4
#ifndef CONFIG_SUN4
sun4c_flush_page_sw
(
pages
+
PAGE_SIZE
);
sun4c_flush_page
(
pages
+
PAGE_SIZE
);
#endif
#endif
}
sun4c_put_pte
(
addr
,
BUCKET_PTE
(
pages
));
sun4c_put_pte
(
addr
,
BUCKET_PTE
(
pages
));
#ifndef CONFIG_SUN4
#ifndef CONFIG_SUN4
...
@@ -1145,32 +1060,7 @@ static struct task_struct *sun4c_alloc_task_struct(void)
...
@@ -1145,32 +1060,7 @@ static struct task_struct *sun4c_alloc_task_struct(void)
return
(
struct
task_struct
*
)
addr
;
return
(
struct
task_struct
*
)
addr
;
}
}
static
void
sun4c_free_task_struct_hw
(
struct
task_struct
*
tsk
)
static
void
sun4c_free_task_struct
(
struct
task_struct
*
tsk
)
{
unsigned
long
tsaddr
=
(
unsigned
long
)
tsk
;
unsigned
long
pages
=
BUCKET_PTE_PAGE
(
sun4c_get_pte
(
tsaddr
));
int
entry
=
BUCKET_NUM
(
tsaddr
);
if
(
atomic_dec_and_test
(
&
(
tsk
)
->
thread
.
refcount
))
{
/* We are deleting a mapping, so the flush here is mandatory. */
sun4c_flush_page_hw
(
tsaddr
);
#ifndef CONFIG_SUN4
sun4c_flush_page_hw
(
tsaddr
+
PAGE_SIZE
);
#endif
sun4c_put_pte
(
tsaddr
,
0
);
#ifndef CONFIG_SUN4
sun4c_put_pte
(
tsaddr
+
PAGE_SIZE
,
0
);
#endif
sun4c_bucket
[
entry
]
=
BUCKET_EMPTY
;
if
(
entry
<
sun4c_lowbucket_avail
)
sun4c_lowbucket_avail
=
entry
;
free_pages
(
pages
,
TASK_STRUCT_ORDER
);
garbage_collect
(
entry
);
}
}
static
void
sun4c_free_task_struct_sw
(
struct
task_struct
*
tsk
)
{
{
unsigned
long
tsaddr
=
(
unsigned
long
)
tsk
;
unsigned
long
tsaddr
=
(
unsigned
long
)
tsk
;
unsigned
long
pages
=
BUCKET_PTE_PAGE
(
sun4c_get_pte
(
tsaddr
));
unsigned
long
pages
=
BUCKET_PTE_PAGE
(
sun4c_get_pte
(
tsaddr
));
...
@@ -1178,9 +1068,9 @@ static void sun4c_free_task_struct_sw(struct task_struct *tsk)
...
@@ -1178,9 +1068,9 @@ static void sun4c_free_task_struct_sw(struct task_struct *tsk)
if
(
atomic_dec_and_test
(
&
(
tsk
)
->
thread
.
refcount
))
{
if
(
atomic_dec_and_test
(
&
(
tsk
)
->
thread
.
refcount
))
{
/* We are deleting a mapping, so the flush here is mandatory. */
/* We are deleting a mapping, so the flush here is mandatory. */
sun4c_flush_page
_sw
(
tsaddr
);
sun4c_flush_page
(
tsaddr
);
#ifndef CONFIG_SUN4
#ifndef CONFIG_SUN4
sun4c_flush_page
_sw
(
tsaddr
+
PAGE_SIZE
);
sun4c_flush_page
(
tsaddr
+
PAGE_SIZE
);
#endif
#endif
sun4c_put_pte
(
tsaddr
,
0
);
sun4c_put_pte
(
tsaddr
,
0
);
#ifndef CONFIG_SUN4
#ifndef CONFIG_SUN4
...
@@ -1452,131 +1342,7 @@ static void sun4c_flush_cache_all(void)
...
@@ -1452,131 +1342,7 @@ static void sun4c_flush_cache_all(void)
}
}
}
}
static
void
sun4c_flush_cache_mm_hw
(
struct
mm_struct
*
mm
)
static
void
sun4c_flush_cache_mm
(
struct
mm_struct
*
mm
)
{
int
new_ctx
=
mm
->
context
;
if
(
new_ctx
!=
NO_CONTEXT
)
{
flush_user_windows
();
if
(
sun4c_context_ring
[
new_ctx
].
num_entries
)
{
struct
sun4c_mmu_entry
*
head
=
&
sun4c_context_ring
[
new_ctx
].
ringhd
;
unsigned
long
flags
;
save_and_cli
(
flags
);
if
(
head
->
next
!=
head
)
{
struct
sun4c_mmu_entry
*
entry
=
head
->
next
;
int
savectx
=
sun4c_get_context
();
sun4c_set_context
(
new_ctx
);
sun4c_flush_context_hw
();
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
sun4c_user_unmap
(
entry
);
free_user_entry
(
new_ctx
,
entry
);
entry
=
next
;
}
while
(
entry
!=
head
);
sun4c_set_context
(
savectx
);
}
restore_flags
(
flags
);
}
}
}
static
void
sun4c_flush_cache_range_hw
(
struct
vm_area_struct
*
vma
,
unsigned
long
start
,
unsigned
long
end
)
{
struct
mm_struct
*
mm
=
vma
->
vm_mm
;
int
new_ctx
=
mm
->
context
;
if
(
new_ctx
!=
NO_CONTEXT
)
{
struct
sun4c_mmu_entry
*
head
=
&
sun4c_context_ring
[
new_ctx
].
ringhd
;
struct
sun4c_mmu_entry
*
entry
;
unsigned
long
flags
;
flush_user_windows
();
save_and_cli
(
flags
);
/* All user segmap chains are ordered on entry->vaddr. */
for
(
entry
=
head
->
next
;
(
entry
!=
head
)
&&
((
entry
->
vaddr
+
SUN4C_REAL_PGDIR_SIZE
)
<
start
);
entry
=
entry
->
next
)
;
/* Tracing various job mixtures showed that this conditional
* only passes ~35% of the time for most worse case situations,
* therefore we avoid all of this gross overhead ~65% of the time.
*/
if
((
entry
!=
head
)
&&
(
entry
->
vaddr
<
end
))
{
int
octx
=
sun4c_get_context
();
sun4c_set_context
(
new_ctx
);
/* At this point, always, (start >= entry->vaddr) and
* (entry->vaddr < end), once the latter condition
* ceases to hold, or we hit the end of the list, we
* exit the loop. The ordering of all user allocated
* segmaps makes this all work out so beautifully.
*/
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
unsigned
long
realend
;
/* "realstart" is always >= entry->vaddr */
realend
=
entry
->
vaddr
+
SUN4C_REAL_PGDIR_SIZE
;
if
(
end
<
realend
)
realend
=
end
;
if
((
realend
-
entry
->
vaddr
)
<=
(
PAGE_SIZE
<<
3
))
{
unsigned
long
page
=
entry
->
vaddr
;
while
(
page
<
realend
)
{
sun4c_flush_page_hw
(
page
);
page
+=
PAGE_SIZE
;
}
}
else
{
sun4c_flush_segment_hw
(
entry
->
vaddr
);
sun4c_user_unmap
(
entry
);
free_user_entry
(
new_ctx
,
entry
);
}
entry
=
next
;
}
while
((
entry
!=
head
)
&&
(
entry
->
vaddr
<
end
));
sun4c_set_context
(
octx
);
}
restore_flags
(
flags
);
}
}
static
void
sun4c_flush_cache_page_hw
(
struct
vm_area_struct
*
vma
,
unsigned
long
page
)
{
struct
mm_struct
*
mm
=
vma
->
vm_mm
;
int
new_ctx
=
mm
->
context
;
/* Sun4c has no separate I/D caches so cannot optimize for non
* text page flushes.
*/
if
(
new_ctx
!=
NO_CONTEXT
)
{
int
octx
=
sun4c_get_context
();
unsigned
long
flags
;
flush_user_windows
();
save_and_cli
(
flags
);
sun4c_set_context
(
new_ctx
);
sun4c_flush_page_hw
(
page
);
sun4c_set_context
(
octx
);
restore_flags
(
flags
);
}
}
static
void
sun4c_flush_page_to_ram_hw
(
unsigned
long
page
)
{
unsigned
long
flags
;
save_and_cli
(
flags
);
sun4c_flush_page_hw
(
page
);
restore_flags
(
flags
);
}
static
void
sun4c_flush_cache_mm_sw
(
struct
mm_struct
*
mm
)
{
{
int
new_ctx
=
mm
->
context
;
int
new_ctx
=
mm
->
context
;
...
@@ -1593,7 +1359,7 @@ static void sun4c_flush_cache_mm_sw(struct mm_struct *mm)
...
@@ -1593,7 +1359,7 @@ static void sun4c_flush_cache_mm_sw(struct mm_struct *mm)
int
savectx
=
sun4c_get_context
();
int
savectx
=
sun4c_get_context
();
sun4c_set_context
(
new_ctx
);
sun4c_set_context
(
new_ctx
);
sun4c_flush_context
_sw
();
sun4c_flush_context
();
do
{
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
...
@@ -1653,11 +1419,11 @@ static void sun4c_flush_cache_range_sw(struct vm_area_struct *vma, unsigned long
...
@@ -1653,11 +1419,11 @@ static void sun4c_flush_cache_range_sw(struct vm_area_struct *vma, unsigned long
if
((
realend
-
entry
->
vaddr
)
<=
(
PAGE_SIZE
<<
3
))
{
if
((
realend
-
entry
->
vaddr
)
<=
(
PAGE_SIZE
<<
3
))
{
unsigned
long
page
=
entry
->
vaddr
;
unsigned
long
page
=
entry
->
vaddr
;
while
(
page
<
realend
)
{
while
(
page
<
realend
)
{
sun4c_flush_page
_sw
(
page
);
sun4c_flush_page
(
page
);
page
+=
PAGE_SIZE
;
page
+=
PAGE_SIZE
;
}
}
}
else
{
}
else
{
sun4c_flush_segment
_sw
(
entry
->
vaddr
);
sun4c_flush_segment
(
entry
->
vaddr
);
sun4c_user_unmap
(
entry
);
sun4c_user_unmap
(
entry
);
free_user_entry
(
new_ctx
,
entry
);
free_user_entry
(
new_ctx
,
entry
);
}
}
...
@@ -1669,7 +1435,7 @@ static void sun4c_flush_cache_range_sw(struct vm_area_struct *vma, unsigned long
...
@@ -1669,7 +1435,7 @@ static void sun4c_flush_cache_range_sw(struct vm_area_struct *vma, unsigned long
}
}
}
}
static
void
sun4c_flush_cache_page
_sw
(
struct
vm_area_struct
*
vma
,
unsigned
long
page
)
static
void
sun4c_flush_cache_page
(
struct
vm_area_struct
*
vma
,
unsigned
long
page
)
{
{
struct
mm_struct
*
mm
=
vma
->
vm_mm
;
struct
mm_struct
*
mm
=
vma
->
vm_mm
;
int
new_ctx
=
mm
->
context
;
int
new_ctx
=
mm
->
context
;
...
@@ -1684,18 +1450,18 @@ static void sun4c_flush_cache_page_sw(struct vm_area_struct *vma, unsigned long
...
@@ -1684,18 +1450,18 @@ static void sun4c_flush_cache_page_sw(struct vm_area_struct *vma, unsigned long
flush_user_windows
();
flush_user_windows
();
save_and_cli
(
flags
);
save_and_cli
(
flags
);
sun4c_set_context
(
new_ctx
);
sun4c_set_context
(
new_ctx
);
sun4c_flush_page
_sw
(
page
);
sun4c_flush_page
(
page
);
sun4c_set_context
(
octx
);
sun4c_set_context
(
octx
);
restore_flags
(
flags
);
restore_flags
(
flags
);
}
}
}
}
static
void
sun4c_flush_page_to_ram
_sw
(
unsigned
long
page
)
static
void
sun4c_flush_page_to_ram
(
unsigned
long
page
)
{
{
unsigned
long
flags
;
unsigned
long
flags
;
save_and_cli
(
flags
);
save_and_cli
(
flags
);
sun4c_flush_page
_sw
(
page
);
sun4c_flush_page
(
page
);
restore_flags
(
flags
);
restore_flags
(
flags
);
}
}
...
@@ -1723,10 +1489,7 @@ static void sun4c_flush_tlb_all(void)
...
@@ -1723,10 +1489,7 @@ static void sun4c_flush_tlb_all(void)
flush_user_windows
();
flush_user_windows
();
while
(
sun4c_kernel_ring
.
num_entries
)
{
while
(
sun4c_kernel_ring
.
num_entries
)
{
next_entry
=
this_entry
->
next
;
next_entry
=
this_entry
->
next
;
if
(
sun4c_vacinfo
.
do_hwflushes
)
sun4c_flush_segment
(
this_entry
->
vaddr
);
sun4c_flush_segment_hw
(
this_entry
->
vaddr
);
else
sun4c_flush_segment_sw
(
this_entry
->
vaddr
);
for
(
ctx
=
0
;
ctx
<
num_contexts
;
ctx
++
)
{
for
(
ctx
=
0
;
ctx
<
num_contexts
;
ctx
++
)
{
sun4c_set_context
(
ctx
);
sun4c_set_context
(
ctx
);
sun4c_put_segmap
(
this_entry
->
vaddr
,
invalid_segment
);
sun4c_put_segmap
(
this_entry
->
vaddr
,
invalid_segment
);
...
@@ -1738,7 +1501,7 @@ static void sun4c_flush_tlb_all(void)
...
@@ -1738,7 +1501,7 @@ static void sun4c_flush_tlb_all(void)
restore_flags
(
flags
);
restore_flags
(
flags
);
}
}
static
void
sun4c_flush_tlb_mm
_hw
(
struct
mm_struct
*
mm
)
static
void
sun4c_flush_tlb_mm
(
struct
mm_struct
*
mm
)
{
{
int
new_ctx
=
mm
->
context
;
int
new_ctx
=
mm
->
context
;
...
@@ -1752,91 +1515,7 @@ static void sun4c_flush_tlb_mm_hw(struct mm_struct *mm)
...
@@ -1752,91 +1515,7 @@ static void sun4c_flush_tlb_mm_hw(struct mm_struct *mm)
int
savectx
=
sun4c_get_context
();
int
savectx
=
sun4c_get_context
();
sun4c_set_context
(
new_ctx
);
sun4c_set_context
(
new_ctx
);
sun4c_flush_context_hw
();
sun4c_flush_context
();
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
sun4c_user_unmap
(
entry
);
free_user_entry
(
new_ctx
,
entry
);
entry
=
next
;
}
while
(
entry
!=
head
);
sun4c_set_context
(
savectx
);
}
restore_flags
(
flags
);
}
}
static
void
sun4c_flush_tlb_range_hw
(
struct
vm_area_struct
*
vma
,
unsigned
long
start
,
unsigned
long
end
)
{
struct
mm_struct
*
mm
=
vma
->
vm_mm
;
int
new_ctx
=
mm
->
context
;
if
(
new_ctx
!=
NO_CONTEXT
)
{
struct
sun4c_mmu_entry
*
head
=
&
sun4c_context_ring
[
new_ctx
].
ringhd
;
struct
sun4c_mmu_entry
*
entry
;
unsigned
long
flags
;
save_and_cli
(
flags
);
/* See commentary in sun4c_flush_cache_range_*(). */
for
(
entry
=
head
->
next
;
(
entry
!=
head
)
&&
((
entry
->
vaddr
+
SUN4C_REAL_PGDIR_SIZE
)
<
start
);
entry
=
entry
->
next
)
;
if
((
entry
!=
head
)
&&
(
entry
->
vaddr
<
end
))
{
int
octx
=
sun4c_get_context
();
sun4c_set_context
(
new_ctx
);
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
sun4c_flush_segment_hw
(
entry
->
vaddr
);
sun4c_user_unmap
(
entry
);
free_user_entry
(
new_ctx
,
entry
);
entry
=
next
;
}
while
((
entry
!=
head
)
&&
(
entry
->
vaddr
<
end
));
sun4c_set_context
(
octx
);
}
restore_flags
(
flags
);
}
}
static
void
sun4c_flush_tlb_page_hw
(
struct
vm_area_struct
*
vma
,
unsigned
long
page
)
{
struct
mm_struct
*
mm
=
vma
->
vm_mm
;
int
new_ctx
=
mm
->
context
;
if
(
new_ctx
!=
NO_CONTEXT
)
{
int
savectx
=
sun4c_get_context
();
unsigned
long
flags
;
save_and_cli
(
flags
);
sun4c_set_context
(
new_ctx
);
page
&=
PAGE_MASK
;
sun4c_flush_page_hw
(
page
);
sun4c_put_pte
(
page
,
0
);
sun4c_set_context
(
savectx
);
restore_flags
(
flags
);
}
}
static
void
sun4c_flush_tlb_mm_sw
(
struct
mm_struct
*
mm
)
{
int
new_ctx
=
mm
->
context
;
if
(
new_ctx
!=
NO_CONTEXT
)
{
struct
sun4c_mmu_entry
*
head
=
&
sun4c_context_ring
[
new_ctx
].
ringhd
;
unsigned
long
flags
;
save_and_cli
(
flags
);
if
(
head
->
next
!=
head
)
{
struct
sun4c_mmu_entry
*
entry
=
head
->
next
;
int
savectx
=
sun4c_get_context
();
sun4c_set_context
(
new_ctx
);
sun4c_flush_context_sw
();
do
{
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
...
@@ -1862,7 +1541,7 @@ static void sun4c_flush_tlb_range_sw(struct vm_area_struct *vma, unsigned long s
...
@@ -1862,7 +1541,7 @@ static void sun4c_flush_tlb_range_sw(struct vm_area_struct *vma, unsigned long s
unsigned
long
flags
;
unsigned
long
flags
;
save_and_cli
(
flags
);
save_and_cli
(
flags
);
/* See commentary in sun4c_flush_cache_range
_*
(). */
/* See commentary in sun4c_flush_cache_range(). */
for
(
entry
=
head
->
next
;
for
(
entry
=
head
->
next
;
(
entry
!=
head
)
&&
((
entry
->
vaddr
+
SUN4C_REAL_PGDIR_SIZE
)
<
start
);
(
entry
!=
head
)
&&
((
entry
->
vaddr
+
SUN4C_REAL_PGDIR_SIZE
)
<
start
);
entry
=
entry
->
next
)
entry
=
entry
->
next
)
...
@@ -1875,7 +1554,7 @@ static void sun4c_flush_tlb_range_sw(struct vm_area_struct *vma, unsigned long s
...
@@ -1875,7 +1554,7 @@ static void sun4c_flush_tlb_range_sw(struct vm_area_struct *vma, unsigned long s
do
{
do
{
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
struct
sun4c_mmu_entry
*
next
=
entry
->
next
;
sun4c_flush_segment
_sw
(
entry
->
vaddr
);
sun4c_flush_segment
(
entry
->
vaddr
);
sun4c_user_unmap
(
entry
);
sun4c_user_unmap
(
entry
);
free_user_entry
(
new_ctx
,
entry
);
free_user_entry
(
new_ctx
,
entry
);
...
@@ -1887,7 +1566,7 @@ static void sun4c_flush_tlb_range_sw(struct vm_area_struct *vma, unsigned long s
...
@@ -1887,7 +1566,7 @@ static void sun4c_flush_tlb_range_sw(struct vm_area_struct *vma, unsigned long s
}
}
}
}
static
void
sun4c_flush_tlb_page
_sw
(
struct
vm_area_struct
*
vma
,
unsigned
long
page
)
static
void
sun4c_flush_tlb_page
(
struct
vm_area_struct
*
vma
,
unsigned
long
page
)
{
{
struct
mm_struct
*
mm
=
vma
->
vm_mm
;
struct
mm_struct
*
mm
=
vma
->
vm_mm
;
int
new_ctx
=
mm
->
context
;
int
new_ctx
=
mm
->
context
;
...
@@ -1899,7 +1578,7 @@ static void sun4c_flush_tlb_page_sw(struct vm_area_struct *vma, unsigned long pa
...
@@ -1899,7 +1578,7 @@ static void sun4c_flush_tlb_page_sw(struct vm_area_struct *vma, unsigned long pa
save_and_cli
(
flags
);
save_and_cli
(
flags
);
sun4c_set_context
(
new_ctx
);
sun4c_set_context
(
new_ctx
);
page
&=
PAGE_MASK
;
page
&=
PAGE_MASK
;
sun4c_flush_page
_sw
(
page
);
sun4c_flush_page
(
page
);
sun4c_put_pte
(
page
,
0
);
sun4c_put_pte
(
page
,
0
);
sun4c_set_context
(
savectx
);
sun4c_set_context
(
savectx
);
restore_flags
(
flags
);
restore_flags
(
flags
);
...
@@ -1923,7 +1602,7 @@ void sun4c_unmapioaddr(unsigned long virt_addr)
...
@@ -1923,7 +1602,7 @@ void sun4c_unmapioaddr(unsigned long virt_addr)
sun4c_put_pte
(
virt_addr
,
0
);
sun4c_put_pte
(
virt_addr
,
0
);
}
}
static
void
sun4c_alloc_context
_hw
(
struct
mm_struct
*
old_mm
,
struct
mm_struct
*
mm
)
static
void
sun4c_alloc_context
(
struct
mm_struct
*
old_mm
,
struct
mm_struct
*
mm
)
{
{
struct
ctx_list
*
ctxp
;
struct
ctx_list
*
ctxp
;
...
@@ -1943,92 +1622,35 @@ static void sun4c_alloc_context_hw(struct mm_struct *old_mm, struct mm_struct *m
...
@@ -1943,92 +1622,35 @@ static void sun4c_alloc_context_hw(struct mm_struct *old_mm, struct mm_struct *m
ctxp
->
ctx_mm
->
context
=
NO_CONTEXT
;
ctxp
->
ctx_mm
->
context
=
NO_CONTEXT
;
ctxp
->
ctx_mm
=
mm
;
ctxp
->
ctx_mm
=
mm
;
mm
->
context
=
ctxp
->
ctx_number
;
mm
->
context
=
ctxp
->
ctx_number
;
sun4c_demap_context_hw
(
&
sun4c_context_ring
[
ctxp
->
ctx_number
],
sun4c_demap_context
(
&
sun4c_context_ring
[
ctxp
->
ctx_number
],
ctxp
->
ctx_number
);
}
/* Switch the current MM context. */
static
void
sun4c_switch_mm_hw
(
struct
mm_struct
*
old_mm
,
struct
mm_struct
*
mm
,
struct
task_struct
*
tsk
,
int
cpu
)
{
struct
ctx_list
*
ctx
;
int
dirty
=
0
;
if
(
mm
->
context
==
NO_CONTEXT
)
{
dirty
=
1
;
sun4c_alloc_context_hw
(
old_mm
,
mm
);
}
else
{
/* Update the LRU ring of contexts. */
ctx
=
ctx_list_pool
+
mm
->
context
;
remove_from_ctx_list
(
ctx
);
add_to_used_ctxlist
(
ctx
);
}
if
(
dirty
||
old_mm
!=
mm
)
sun4c_set_context
(
mm
->
context
);
}
static
void
sun4c_destroy_context_hw
(
struct
mm_struct
*
mm
)
{
struct
ctx_list
*
ctx_old
;
if
(
mm
->
context
!=
NO_CONTEXT
)
{
sun4c_demap_context_hw
(
&
sun4c_context_ring
[
mm
->
context
],
mm
->
context
);
ctx_old
=
ctx_list_pool
+
mm
->
context
;
remove_from_ctx_list
(
ctx_old
);
add_to_free_ctxlist
(
ctx_old
);
mm
->
context
=
NO_CONTEXT
;
}
}
static
void
sun4c_alloc_context_sw
(
struct
mm_struct
*
old_mm
,
struct
mm_struct
*
mm
)
{
struct
ctx_list
*
ctxp
;
ctxp
=
ctx_free
.
next
;
if
(
ctxp
!=
&
ctx_free
)
{
remove_from_ctx_list
(
ctxp
);
add_to_used_ctxlist
(
ctxp
);
mm
->
context
=
ctxp
->
ctx_number
;
ctxp
->
ctx_mm
=
mm
;
return
;
}
ctxp
=
ctx_used
.
next
;
if
(
ctxp
->
ctx_mm
==
old_mm
)
ctxp
=
ctxp
->
next
;
remove_from_ctx_list
(
ctxp
);
add_to_used_ctxlist
(
ctxp
);
ctxp
->
ctx_mm
->
context
=
NO_CONTEXT
;
ctxp
->
ctx_mm
=
mm
;
mm
->
context
=
ctxp
->
ctx_number
;
sun4c_demap_context_sw
(
&
sun4c_context_ring
[
ctxp
->
ctx_number
],
ctxp
->
ctx_number
);
ctxp
->
ctx_number
);
}
}
/* Switch the current MM context. */
/* Switch the current MM context. */
static
void
sun4c_switch_mm
_sw
(
struct
mm_struct
*
old_mm
,
struct
mm_struct
*
mm
,
struct
task_struct
*
tsk
,
int
cpu
)
static
void
sun4c_switch_mm
(
struct
mm_struct
*
old_mm
,
struct
mm_struct
*
mm
,
struct
task_struct
*
tsk
,
int
cpu
)
{
{
struct
ctx_list
*
ctx
;
struct
ctx_list
*
ctx
;
int
dirty
=
0
;
int
dirty
=
0
;
if
(
mm
->
context
==
NO_CONTEXT
)
{
if
(
mm
->
context
==
NO_CONTEXT
)
{
dirty
=
1
;
dirty
=
1
;
sun4c_alloc_context
_sw
(
old_mm
,
mm
);
sun4c_alloc_context
(
old_mm
,
mm
);
}
else
{
}
else
{
/* Update the LRU ring of contexts. */
/* Update the LRU ring of contexts. */
ctx
=
ctx_list_pool
+
mm
->
context
;
ctx
=
ctx_list_pool
+
mm
->
context
;
remove_from_ctx_list
(
ctx
);
remove_from_ctx_list
(
ctx
);
add_to_used_ctxlist
(
ctx
);
add_to_used_ctxlist
(
ctx
);
}
}
if
(
dirty
||
old_mm
!=
mm
)
if
(
dirty
||
old_mm
!=
mm
)
sun4c_set_context
(
mm
->
context
);
sun4c_set_context
(
mm
->
context
);
}
}
static
void
sun4c_destroy_context
_sw
(
struct
mm_struct
*
mm
)
static
void
sun4c_destroy_context
(
struct
mm_struct
*
mm
)
{
{
struct
ctx_list
*
ctx_old
;
struct
ctx_list
*
ctx_old
;
if
(
mm
->
context
!=
NO_CONTEXT
)
{
if
(
mm
->
context
!=
NO_CONTEXT
)
{
sun4c_demap_context
_sw
(
&
sun4c_context_ring
[
mm
->
context
],
mm
->
context
);
sun4c_demap_context
(
&
sun4c_context_ring
[
mm
->
context
],
mm
->
context
);
ctx_old
=
ctx_list_pool
+
mm
->
context
;
ctx_old
=
ctx_list_pool
+
mm
->
context
;
remove_from_ctx_list
(
ctx_old
);
remove_from_ctx_list
(
ctx_old
);
add_to_free_ctxlist
(
ctx_old
);
add_to_free_ctxlist
(
ctx_old
);
...
@@ -2095,7 +1717,7 @@ static void sun4c_pgd_set(pgd_t * pgdp, pmd_t * pmdp)
...
@@ -2095,7 +1717,7 @@ static void sun4c_pgd_set(pgd_t * pgdp, pmd_t * pmdp)
static
void
sun4c_pmd_set
(
pmd_t
*
pmdp
,
pte_t
*
ptep
)
static
void
sun4c_pmd_set
(
pmd_t
*
pmdp
,
pte_t
*
ptep
)
{
{
*
pmdp
=
(
PGD_TABLE
|
(
unsigned
long
)
ptep
);
*
pmdp
=
__pmd
(
PGD_TABLE
|
(
unsigned
long
)
ptep
);
}
}
static
int
sun4c_pte_present
(
pte_t
pte
)
static
int
sun4c_pte_present
(
pte_t
pte
)
...
@@ -2178,10 +1800,7 @@ static inline unsigned long sun4c_pmd_page(pmd_t pmd)
...
@@ -2178,10 +1800,7 @@ static inline unsigned long sun4c_pmd_page(pmd_t pmd)
return
(
pmd_val
(
pmd
)
&
PAGE_MASK
);
return
(
pmd_val
(
pmd
)
&
PAGE_MASK
);
}
}
static
unsigned
long
sun4c_pgd_page
(
pgd_t
pgd
)
static
unsigned
long
sun4c_pgd_page
(
pgd_t
pgd
)
{
return
0
;
}
{
return
0
;
}
/* to find an entry in a page-table-directory */
/* to find an entry in a page-table-directory */
static
inline
pgd_t
*
sun4c_pgd_offset
(
struct
mm_struct
*
mm
,
unsigned
long
address
)
static
inline
pgd_t
*
sun4c_pgd_offset
(
struct
mm_struct
*
mm
,
unsigned
long
address
)
...
@@ -2275,9 +1894,7 @@ static pmd_t *sun4c_pmd_alloc_one_fast(struct mm_struct *mm, unsigned long addre
...
@@ -2275,9 +1894,7 @@ static pmd_t *sun4c_pmd_alloc_one_fast(struct mm_struct *mm, unsigned long addre
return
NULL
;
return
NULL
;
}
}
static
void
sun4c_free_pmd_fast
(
pmd_t
*
pmd
)
static
void
sun4c_free_pmd_fast
(
pmd_t
*
pmd
)
{
}
{
}
static
int
sun4c_check_pgt_cache
(
int
low
,
int
high
)
static
int
sun4c_check_pgt_cache
(
int
low
,
int
high
)
{
{
...
@@ -2470,37 +2087,31 @@ void __init ld_mmu_sun4c(void)
...
@@ -2470,37 +2087,31 @@ void __init ld_mmu_sun4c(void)
_SUN4C_PAGE_IO
|
_SUN4C_PAGE_NOCACHE
;
_SUN4C_PAGE_IO
|
_SUN4C_PAGE_NOCACHE
;
/* Functions */
/* Functions */
#ifndef CONFIG_SMP
BTFIXUPSET_CALL
(
___xchg32
,
___xchg32_sun4c
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
___xchg32
,
___xchg32_sun4c
,
BTFIXUPCALL_NORM
);
#endif
BTFIXUPSET_CALL
(
do_check_pgt_cache
,
sun4c_check_pgt_cache
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
do_check_pgt_cache
,
sun4c_check_pgt_cache
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_all
,
sun4c_flush_cache_all
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_all
,
sun4c_flush_cache_all
,
BTFIXUPCALL_NORM
);
if
(
sun4c_vacinfo
.
do_hwflushes
)
{
if
(
sun4c_vacinfo
.
do_hwflushes
)
{
BTFIXUPSET_CALL
(
flush_cache_mm
,
sun4c_flush_cache_mm_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
sun4c_flush_page
,
sun4c_flush_page_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_range
,
sun4c_flush_cache_range_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
sun4c_flush_segment
,
sun4c_flush_segment_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_page
,
sun4c_flush_cache_page_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
sun4c_flush_context
,
sun4c_flush_context_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
__flush_page_to_ram
,
sun4c_flush_page_to_ram_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_mm
,
sun4c_flush_tlb_mm_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_range
,
sun4c_flush_tlb_range_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_page
,
sun4c_flush_tlb_page_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
free_task_struct
,
sun4c_free_task_struct_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
switch_mm
,
sun4c_switch_mm_hw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
destroy_context
,
sun4c_destroy_context_hw
,
BTFIXUPCALL_NORM
);
}
else
{
}
else
{
BTFIXUPSET_CALL
(
flush_cache_mm
,
sun4c_flush_cache_mm_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
sun4c_flush_page
,
sun4c_flush_page_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_range
,
sun4c_flush_cache_range_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
sun4c_flush_segment
,
sun4c_flush_segment_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_page
,
sun4c_flush_cache_page_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
sun4c_flush_context
,
sun4c_flush_context_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
__flush_page_to_ram
,
sun4c_flush_page_to_ram_sw
,
BTFIXUPCALL_NORM
);
}
BTFIXUPSET_CALL
(
flush_tlb_mm
,
sun4c_flush_tlb_mm_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_range
,
sun4c_flush_tlb_range_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_mm
,
sun4c_flush_tlb_mm
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_page
,
sun4c_flush_tlb_page_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_mm
,
sun4c_flush_cache_mm
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
free_task_struct
,
sun4c_free_task_struct_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
destroy_context
,
sun4c_destroy_context
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
switch_mm
,
sun4c_switch_mm_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
switch_mm
,
sun4c_switch_mm
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
destroy_context
,
sun4c_destroy_context_sw
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_page
,
sun4c_flush_cache_page
,
BTFIXUPCALL_NORM
);
}
BTFIXUPSET_CALL
(
flush_tlb_page
,
sun4c_flush_tlb_page
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_range
,
sun4c_flush_tlb_range
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_cache_range
,
sun4c_flush_cache_range
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
free_task_struct
,
sun4c_free_task_struct
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
__flush_page_to_ram
,
sun4c_flush_page_to_ram
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_all
,
sun4c_flush_tlb_all
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_tlb_all
,
sun4c_flush_tlb_all
,
BTFIXUPCALL_NORM
);
BTFIXUPSET_CALL
(
flush_sig_insns
,
sun4c_flush_sig_insns
,
BTFIXUPCALL_NOP
);
BTFIXUPSET_CALL
(
flush_sig_insns
,
sun4c_flush_sig_insns
,
BTFIXUPCALL_NOP
);
...
...
include/asm-sparc/btfixup.h
View file @
d0ac249b
...
@@ -16,7 +16,22 @@ extern unsigned int ___illegal_use_of_BTFIXUP_SIMM13_in_module(void);
...
@@ -16,7 +16,22 @@ extern unsigned int ___illegal_use_of_BTFIXUP_SIMM13_in_module(void);
extern
unsigned
int
___illegal_use_of_BTFIXUP_SETHI_in_module
(
void
);
extern
unsigned
int
___illegal_use_of_BTFIXUP_SETHI_in_module
(
void
);
extern
unsigned
int
___illegal_use_of_BTFIXUP_HALF_in_module
(
void
);
extern
unsigned
int
___illegal_use_of_BTFIXUP_HALF_in_module
(
void
);
extern
unsigned
int
___illegal_use_of_BTFIXUP_INT_in_module
(
void
);
extern
unsigned
int
___illegal_use_of_BTFIXUP_INT_in_module
(
void
);
#endif
#define BTFIXUP_SIMM13(__name) ___illegal_use_of_BTFIXUP_SIMM13_in_module()
#define BTFIXUP_HALF(__name) ___illegal_use_of_BTFIXUP_HALF_in_module()
#define BTFIXUP_SETHI(__name) ___illegal_use_of_BTFIXUP_SETHI_in_module()
#define BTFIXUP_INT(__name) ___illegal_use_of_BTFIXUP_INT_in_module()
#define BTFIXUP_BLACKBOX(__name) ___illegal_use_of_BTFIXUP_BLACKBOX_in_module
#else
#define BTFIXUP_SIMM13(__name) ___sf_##__name()
#define BTFIXUP_HALF(__name) ___af_##__name()
#define BTFIXUP_SETHI(__name) ___hf_##__name()
#define BTFIXUP_INT(__name) ((unsigned int)&___i_##__name)
/* This must be written in assembly and present in a sethi */
#define BTFIXUP_BLACKBOX(__name) ___b_##__name
#endif
/* MODULE */
/* Fixup call xx */
/* Fixup call xx */
...
@@ -30,12 +45,6 @@ extern unsigned int ___illegal_use_of_BTFIXUP_INT_in_module(void);
...
@@ -30,12 +45,6 @@ extern unsigned int ___illegal_use_of_BTFIXUP_INT_in_module(void);
#define BTFIXUPDEF_BLACKBOX(__name) \
#define BTFIXUPDEF_BLACKBOX(__name) \
extern unsigned ___bs_##__name[2];
extern unsigned ___bs_##__name[2];
#ifdef MODULE
#define BTFIXUP_BLACKBOX(__name) ___illegal_use_of_BTFIXUP_BLACKBOX_in_module
#else
/* This must be written in assembly and present in a sethi */
#define BTFIXUP_BLACKBOX(__name) ___b_##__name
#endif
/* Put bottom 13bits into some register variable */
/* Put bottom 13bits into some register variable */
...
@@ -55,11 +64,6 @@ extern unsigned int ___illegal_use_of_BTFIXUP_INT_in_module(void);
...
@@ -55,11 +64,6 @@ extern unsigned int ___illegal_use_of_BTFIXUP_INT_in_module(void);
__asm__ ("or %%g0, ___s_" #__name "__btset_" #__val ", %0" : "=r"(ret));\
__asm__ ("or %%g0, ___s_" #__name "__btset_" #__val ", %0" : "=r"(ret));\
return ret; \
return ret; \
}
}
#ifdef MODULE
#define BTFIXUP_SIMM13(__name) ___illegal_use_of_BTFIXUP_SIMM13_in_module()
#else
#define BTFIXUP_SIMM13(__name) ___sf_##__name()
#endif
/* Put either bottom 13 bits, or upper 22 bits into some register variable
/* Put either bottom 13 bits, or upper 22 bits into some register variable
* (depending on the value, this will lead into sethi FIX, reg; or
* (depending on the value, this will lead into sethi FIX, reg; or
...
@@ -82,11 +86,6 @@ extern unsigned int ___illegal_use_of_BTFIXUP_INT_in_module(void);
...
@@ -82,11 +86,6 @@ extern unsigned int ___illegal_use_of_BTFIXUP_INT_in_module(void);
__asm__ ("or %%g0, ___a_" #__name "__btset_" #__val ", %0" : "=r"(ret));\
__asm__ ("or %%g0, ___a_" #__name "__btset_" #__val ", %0" : "=r"(ret));\
return ret; \
return ret; \
}
}
#ifdef MODULE
#define BTFIXUP_HALF(__name) ___illegal_use_of_BTFIXUP_HALF_in_module()
#else
#define BTFIXUP_HALF(__name) ___af_##__name()
#endif
/* Put upper 22 bits into some register variable */
/* Put upper 22 bits into some register variable */
...
@@ -107,22 +106,12 @@ extern unsigned int ___illegal_use_of_BTFIXUP_INT_in_module(void);
...
@@ -107,22 +106,12 @@ extern unsigned int ___illegal_use_of_BTFIXUP_INT_in_module(void);
"=r"(ret)); \
"=r"(ret)); \
return ret; \
return ret; \
}
}
#ifdef MODULE
#define BTFIXUP_SETHI(__name) ___illegal_use_of_BTFIXUP_SETHI_in_module()
#else
#define BTFIXUP_SETHI(__name) ___hf_##__name()
#endif
/* Put a full 32bit integer into some register variable */
/* Put a full 32bit integer into some register variable */
#define BTFIXUPDEF_INT(__name) \
#define BTFIXUPDEF_INT(__name) \
extern unsigned char ___i_##__name; \
extern unsigned char ___i_##__name; \
extern unsigned ___is_##__name[2];
extern unsigned ___is_##__name[2];
#ifdef MODULE
#define BTFIXUP_INT(__name) ___illegal_use_of_BTFIXUP_INT_in_module()
#else
#define BTFIXUP_INT(__name) ((unsigned int)&___i_##__name)
#endif
#define BTFIXUPCALL_NORM 0x00000000
/* Always call */
#define BTFIXUPCALL_NORM 0x00000000
/* Always call */
#define BTFIXUPCALL_NOP 0x01000000
/* Possibly optimize to nop */
#define BTFIXUPCALL_NOP 0x01000000
/* Possibly optimize to nop */
...
...
include/asm-sparc/ultra.h
deleted
100644 → 0
View file @
a9907091
/* $Id: ultra.h,v 1.2 1995/11/25 02:33:10 davem Exp $
* ultra.h: Definitions and defines for the TI V9 UltraSparc.
*
* Copyright (C) 1995 David S. Miller (davem@caip.rutgers.edu)
*/
#ifndef _SPARC_ULTRA_H
#define _SPARC_ULTRA_H
/* Spitfire MMU control register:
*
* ----------------------------------------------------------
* | | IMPL | VERS | | MID | |
* ----------------------------------------------------------
* 64 31-28 27-24 23-22 21-17 16 0
*
* IMPL: Implementation of this Spitfire.
* VERS: Version of this Spitfire.
* MID: Module ID of this processor.
*/
#define SPITFIRE_MIDMASK 0x00000000003e0000
/* Spitfire Load Store Unit control register:
*
* ---------------------------------------------------------------------
* | RSV | PWR | PWW | VWR | VWW | RSV | PMASK | DME | IME | DCE | ICE |
* ---------------------------------------------------------------------
* 63-25 24 23 22 21 20 19-4 3 2 1 0
*
* PWR: Physical Watchpoint Read enable: 0=off 1=on
* PWW: Physical Watchpoint Write enable: 0=off 1=on
* VWR: Virtual Watchpoint Read enable: 0=off 1=on
* VWW: Virtual Watchpoint Write enable: 0=off 1=on
* PMASK: Parity MASK ???
* DME: Data MMU Enable: 0=off 1=on
* IME: Instruction MMU Enable: 0=off 1=on
* DCE: Data Cache Enable: 0=off 1=on
* ICE: Instruction Cache Enable: 0=off 1=on
*/
#define SPITFIRE_LSU_PWR 0x01000000
#define SPITFIRE_LSU_PWW 0x00800000
#define SPITFIRE_LSU_VWR 0x00400000
#define SPITFIRE_LSU_VWW 0x00200000
#define SPITFIRE_LSU_PMASK 0x000ffff0
#define SPITFIRE_LSU_DME 0x00000008
#define SPITFIRE_LSU_IME 0x00000004
#define SPITFIRE_LSU_DCE 0x00000002
#define SPITFIRE_LSU_ICE 0x00000001
#endif
/* !(_SPARC_ULTRA_H) */
include/asm-sparc/vac-ops.h
View file @
d0ac249b
...
@@ -107,8 +107,6 @@ struct sun4c_vac_props {
...
@@ -107,8 +107,6 @@ struct sun4c_vac_props {
extern
struct
sun4c_vac_props
sun4c_vacinfo
;
extern
struct
sun4c_vac_props
sun4c_vacinfo
;
extern
void
sun4c_flush_all
(
void
);
/* sun4c_enable_vac() enables the sun4c virtual address cache. */
/* sun4c_enable_vac() enables the sun4c virtual address cache. */
extern
__inline__
void
sun4c_enable_vac
(
void
)
extern
__inline__
void
sun4c_enable_vac
(
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