Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
fc0c49f5
Commit
fc0c49f5
authored
Mar 29, 2012
by
Chris Metcalf
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
arch/tile: support kexec() for tilegx
Signed-off-by:
Chris Metcalf
<
cmetcalf@tilera.com
>
parent
cd6f32aa
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
300 additions
and
9 deletions
+300
-9
arch/tile/include/asm/kexec.h
arch/tile/include/asm/kexec.h
+12
-0
arch/tile/kernel/Makefile
arch/tile/kernel/Makefile
+1
-1
arch/tile/kernel/machine_kexec.c
arch/tile/kernel/machine_kexec.c
+27
-8
arch/tile/kernel/relocate_kernel_32.S
arch/tile/kernel/relocate_kernel_32.S
+0
-0
arch/tile/kernel/relocate_kernel_64.S
arch/tile/kernel/relocate_kernel_64.S
+260
-0
No files found.
arch/tile/include/asm/kexec.h
View file @
fc0c49f5
...
@@ -19,12 +19,24 @@
...
@@ -19,12 +19,24 @@
#include <asm/page.h>
#include <asm/page.h>
#ifndef __tilegx__
/* Maximum physical address we can use pages from. */
/* Maximum physical address we can use pages from. */
#define KEXEC_SOURCE_MEMORY_LIMIT TASK_SIZE
#define KEXEC_SOURCE_MEMORY_LIMIT TASK_SIZE
/* Maximum address we can reach in physical address mode. */
/* Maximum address we can reach in physical address mode. */
#define KEXEC_DESTINATION_MEMORY_LIMIT TASK_SIZE
#define KEXEC_DESTINATION_MEMORY_LIMIT TASK_SIZE
/* Maximum address we can use for the control code buffer. */
/* Maximum address we can use for the control code buffer. */
#define KEXEC_CONTROL_MEMORY_LIMIT TASK_SIZE
#define KEXEC_CONTROL_MEMORY_LIMIT TASK_SIZE
#else
/* We need to limit the memory below PGDIR_SIZE since
* we only setup page table for [0, PGDIR_SIZE) before final kexec.
*/
/* Maximum physical address we can use pages from. */
#define KEXEC_SOURCE_MEMORY_LIMIT PGDIR_SIZE
/* Maximum address we can reach in physical address mode. */
#define KEXEC_DESTINATION_MEMORY_LIMIT PGDIR_SIZE
/* Maximum address we can use for the control code buffer. */
#define KEXEC_CONTROL_MEMORY_LIMIT PGDIR_SIZE
#endif
#define KEXEC_CONTROL_PAGE_SIZE PAGE_SIZE
#define KEXEC_CONTROL_PAGE_SIZE PAGE_SIZE
...
...
arch/tile/kernel/Makefile
View file @
fc0c49f5
...
@@ -13,5 +13,5 @@ obj-$(CONFIG_COMPAT) += compat.o compat_signal.o
...
@@ -13,5 +13,5 @@ obj-$(CONFIG_COMPAT) += compat.o compat_signal.o
obj-$(CONFIG_SMP)
+=
smpboot.o smp.o tlb.o
obj-$(CONFIG_SMP)
+=
smpboot.o smp.o tlb.o
obj-$(CONFIG_MODULES)
+=
module.o
obj-$(CONFIG_MODULES)
+=
module.o
obj-$(CONFIG_EARLY_PRINTK)
+=
early_printk.o
obj-$(CONFIG_EARLY_PRINTK)
+=
early_printk.o
obj-$(CONFIG_KEXEC)
+=
machine_kexec.o relocate_kernel.o
obj-$(CONFIG_KEXEC)
+=
machine_kexec.o relocate_kernel
_
$(BITS)
.o
obj-$(CONFIG_PCI)
+=
pci.o
obj-$(CONFIG_PCI)
+=
pci.o
arch/tile/kernel/machine_kexec.c
View file @
fc0c49f5
...
@@ -31,6 +31,8 @@
...
@@ -31,6 +31,8 @@
#include <asm/pgalloc.h>
#include <asm/pgalloc.h>
#include <asm/cacheflush.h>
#include <asm/cacheflush.h>
#include <asm/checksum.h>
#include <asm/checksum.h>
#include <asm/tlbflush.h>
#include <asm/homecache.h>
#include <hv/hypervisor.h>
#include <hv/hypervisor.h>
...
@@ -222,11 +224,22 @@ struct page *kimage_alloc_pages_arch(gfp_t gfp_mask, unsigned int order)
...
@@ -222,11 +224,22 @@ struct page *kimage_alloc_pages_arch(gfp_t gfp_mask, unsigned int order)
return
alloc_pages_node
(
0
,
gfp_mask
,
order
);
return
alloc_pages_node
(
0
,
gfp_mask
,
order
);
}
}
/*
* Address range in which pa=va mapping is set in setup_quasi_va_is_pa().
* For tilepro, PAGE_OFFSET is used since this is the largest possbile value
* for tilepro, while for tilegx, we limit it to entire middle level page
* table which we assume has been allocated and is undoubtedly large enough.
*/
#ifndef __tilegx__
#define QUASI_VA_IS_PA_ADDR_RANGE PAGE_OFFSET
#else
#define QUASI_VA_IS_PA_ADDR_RANGE PGDIR_SIZE
#endif
static
void
setup_quasi_va_is_pa
(
void
)
static
void
setup_quasi_va_is_pa
(
void
)
{
{
HV_PTE
*
pgtable
;
HV_PTE
pte
;
HV_PTE
pte
;
int
i
;
unsigned
long
i
;
/*
/*
* Flush our TLB to prevent conflicts between the previous contents
* Flush our TLB to prevent conflicts between the previous contents
...
@@ -234,16 +247,22 @@ static void setup_quasi_va_is_pa(void)
...
@@ -234,16 +247,22 @@ static void setup_quasi_va_is_pa(void)
*/
*/
local_flush_tlb_all
();
local_flush_tlb_all
();
/* setup VA is PA, at least up to PAGE_OFFSET */
/*
* setup VA is PA, at least up to QUASI_VA_IS_PA_ADDR_RANGE.
pgtable
=
(
HV_PTE
*
)
current
->
mm
->
pgd
;
* Note here we assume that level-1 page table is defined by
* HPAGE_SIZE.
*/
pte
=
hv_pte
(
_PAGE_KERNEL
|
_PAGE_HUGE_PAGE
);
pte
=
hv_pte
(
_PAGE_KERNEL
|
_PAGE_HUGE_PAGE
);
pte
=
hv_pte_set_mode
(
pte
,
HV_PTE_MODE_CACHE_NO_L3
);
pte
=
hv_pte_set_mode
(
pte
,
HV_PTE_MODE_CACHE_NO_L3
);
for
(
i
=
0
;
i
<
(
QUASI_VA_IS_PA_ADDR_RANGE
>>
HPAGE_SHIFT
);
i
++
)
{
for
(
i
=
0
;
i
<
pgd_index
(
PAGE_OFFSET
);
i
++
)
{
unsigned
long
vaddr
=
i
<<
HPAGE_SHIFT
;
pgd_t
*
pgd
=
pgd_offset
(
current
->
mm
,
vaddr
);
pud_t
*
pud
=
pud_offset
(
pgd
,
vaddr
);
pte_t
*
ptep
=
(
pte_t
*
)
pmd_offset
(
pud
,
vaddr
);
unsigned
long
pfn
=
i
<<
(
HPAGE_SHIFT
-
PAGE_SHIFT
);
unsigned
long
pfn
=
i
<<
(
HPAGE_SHIFT
-
PAGE_SHIFT
);
if
(
pfn_valid
(
pfn
))
if
(
pfn_valid
(
pfn
))
__set_pte
(
&
pgtable
[
i
]
,
pfn_pte
(
pfn
,
pte
));
__set_pte
(
ptep
,
pfn_pte
(
pfn
,
pte
));
}
}
}
}
...
...
arch/tile/kernel/relocate_kernel.S
→
arch/tile/kernel/relocate_kernel
_32
.S
View file @
fc0c49f5
File moved
arch/tile/kernel/relocate_kernel_64.S
0 → 100644
View file @
fc0c49f5
/*
*
Copyright
2011
Tilera
Corporation
.
All
Rights
Reserved
.
*
*
This
program
is
free
software
; you can redistribute it and/or
*
modify
it
under
the
terms
of
the
GNU
General
Public
License
*
as
published
by
the
Free
Software
Foundation
,
version
2
.
*
*
This
program
is
distributed
in
the
hope
that
it
will
be
useful
,
but
*
WITHOUT
ANY
WARRANTY
; without even the implied warranty of
*
MERCHANTABILITY
OR
FITNESS
FOR
A
PARTICULAR
PURPOSE
,
GOOD
TITLE
or
*
NON
INFRINGEMENT
.
See
the
GNU
General
Public
License
for
*
more
details
.
*
*
copy
new
kernel
into
place
and
then
call
hv_reexec
*
*/
#include <linux/linkage.h>
#include <arch/chip.h>
#include <asm/page.h>
#include <hv/hypervisor.h>
#undef RELOCATE_NEW_KERNEL_VERBOSE
STD_ENTRY
(
relocate_new_kernel
)
move
r30
,
r0
/*
page
list
*/
move
r31
,
r1
/*
address
of
page
we
are
on
*/
move
r32
,
r2
/*
start
address
of
new
kernel
*/
shrui
r1
,
r1
,
PAGE_SHIFT
addi
r1
,
r1
,
1
shli
sp
,
r1
,
PAGE_SHIFT
addi
sp
,
sp
,
-
8
/
*
we
now
have
a
stack
(
whether
we
need
one
or
not
)
*/
moveli
r40
,
hw2_last
(
hv_console_putc
)
shl16insli
r40
,
r40
,
hw1
(
hv_console_putc
)
shl16insli
r40
,
r40
,
hw0
(
hv_console_putc
)
#ifdef RELOCATE_NEW_KERNEL_VERBOSE
moveli
r0
,
'r'
jalr
r40
moveli
r0
,
'_'
jalr
r40
moveli
r0
,
'n'
jalr
r40
moveli
r0
,
'_'
jalr
r40
moveli
r0
,
'k'
jalr
r40
moveli
r0
,
'\n'
jalr
r40
#endif
/
*
*
Throughout
this
code
r30
is
pointer
to
the
element
of
page
*
list
we
are
working
on
.
*
*
Normally
we
get
to
the
next
element
of
the
page
list
by
*
incrementing
r30
by
eight
.
The
exception
is
if
the
element
*
on
the
page
list
is
an
IND_INDIRECTION
in
which
case
we
use
*
the
element
with
the
low
bits
masked
off
as
the
new
value
*
of
r30
.
*
*
To
get
this
started
,
we
need
the
value
passed
to
us
(
which
*
will
always
be
an
IND_INDIRECTION
)
in
memory
somewhere
with
*
r30
pointing
at
it
.
To
do
that
,
we
push
the
value
passed
*
to
us
on
the
stack
and
make
r30
point
to
it
.
*/
st
sp
,
r30
move
r30
,
sp
addi
sp
,
sp
,
-
16
#if CHIP_HAS_CBOX_HOME_MAP()
/
*
*
On
TILE
-
GX
,
we
need
to
flush
all
tiles
' caches, since we may
*
have
been
doing
hash
-
for
-
home
caching
there
.
Note
that
we
*
must
do
this
_after_
we
're completely done modifying any memory
*
other
than
our
output
buffer
(
which
we
know
is
locally
cached
)
.
*
We
want
the
caches
to
be
fully
clean
when
we
do
the
reexec
,
*
because
the
hypervisor
is
going
to
do
this
flush
again
at
that
*
point
,
and
we
don
't want that second flush to overwrite any memory.
*/
{
move
r0
,
zero
/*
cache_pa
*/
moveli
r1
,
hw2_last
(
HV_FLUSH_EVICT_L2
)
}
{
shl16insli
r1
,
r1
,
hw1
(
HV_FLUSH_EVICT_L2
)
movei
r2
,
-
1
/*
cache_cpumask
; -1 means all client tiles */
}
{
shl16insli
r1
,
r1
,
hw0
(
HV_FLUSH_EVICT_L2
)
/*
cache_control
*/
move
r3
,
zero
/*
tlb_va
*/
}
{
move
r4
,
zero
/*
tlb_length
*/
move
r5
,
zero
/*
tlb_pgsize
*/
}
{
move
r6
,
zero
/*
tlb_cpumask
*/
move
r7
,
zero
/*
asids
*/
}
{
moveli
r20
,
hw2_last
(
hv_flush_remote
)
move
r8
,
zero
/*
asidcount
*/
}
shl16insli
r20
,
r20
,
hw1
(
hv_flush_remote
)
shl16insli
r20
,
r20
,
hw0
(
hv_flush_remote
)
jalr
r20
#endif
/
*
r33
is
destination
pointer
,
default
to
zero
*/
moveli
r33
,
0
.
Lloop
:
ld
r10
,
r30
andi
r9
,
r10
,
0xf
/*
low
4
bits
tell
us
what
type
it
is
*/
xor
r10
,
r10
,
r9
/*
r10
is
now
value
with
low
4
bits
stripped
*/
cmpeqi
r0
,
r9
,
0x1
/*
IND_DESTINATION
*/
beqzt
r0
,
.
Ltry2
move
r33
,
r10
#ifdef RELOCATE_NEW_KERNEL_VERBOSE
moveli
r0
,
'd'
jalr
r40
#endif
addi
r30
,
r30
,
8
j
.
Lloop
.
Ltry2
:
cmpeqi
r0
,
r9
,
0x2
/*
IND_INDIRECTION
*/
beqzt
r0
,
.
Ltry4
move
r30
,
r10
#ifdef RELOCATE_NEW_KERNEL_VERBOSE
moveli
r0
,
'i'
jalr
r40
#endif
j
.
Lloop
.
Ltry4
:
cmpeqi
r0
,
r9
,
0x4
/*
IND_DONE
*/
beqzt
r0
,
.
Ltry8
mf
#ifdef RELOCATE_NEW_KERNEL_VERBOSE
moveli
r0
,
'D'
jalr
r40
moveli
r0
,
'\n'
jalr
r40
#endif
move
r0
,
r32
moveli
r41
,
hw2_last
(
hv_reexec
)
shl16insli
r41
,
r41
,
hw1
(
hv_reexec
)
shl16insli
r41
,
r41
,
hw0
(
hv_reexec
)
jalr
r41
/
*
we
should
not
get
here
*/
moveli
r0
,
'?'
jalr
r40
moveli
r0
,
'\n'
jalr
r40
j
.
Lhalt
.
Ltry8
:
cmpeqi
r0
,
r9
,
0x8
/*
IND_SOURCE
*/
beqz
r0
,
.
Lerr
/*
unknown
type
*/
/
*
copy
page
at
r10
to
page
at
r33
*/
move
r11
,
r33
moveli
r0
,
hw2_last
(
PAGE_SIZE
)
shl16insli
r0
,
r0
,
hw1
(
PAGE_SIZE
)
shl16insli
r0
,
r0
,
hw0
(
PAGE_SIZE
)
add
r33
,
r33
,
r0
/
*
copy
word
at
r10
to
word
at
r11
until
r11
equals
r33
*/
/
*
We
know
page
size
must
be
multiple
of
8
,
so
we
can
unroll
*
8
times
safely
without
any
edge
case
checking
.
*
*
Issue
a
flush
of
the
destination
every
8
words
to
avoid
*
incoherence
when
starting
the
new
kernel
.
(
Now
this
is
*
just
good
paranoia
because
the
hv_reexec
call
will
also
*
take
care
of
this
.
)
*/
1
:
{
ld
r0
,
r10
; addi r10, r10, 8 }
{
st
r11
,
r0
; addi r11, r11, 8 }
{
ld
r0
,
r10
; addi r10, r10, 8 }
{
st
r11
,
r0
; addi r11, r11, 8 }
{
ld
r0
,
r10
; addi r10, r10, 8 }
{
st
r11
,
r0
; addi r11, r11, 8 }
{
ld
r0
,
r10
; addi r10, r10, 8 }
{
st
r11
,
r0
; addi r11, r11, 8 }
{
ld
r0
,
r10
; addi r10, r10, 8 }
{
st
r11
,
r0
; addi r11, r11, 8 }
{
ld
r0
,
r10
; addi r10, r10, 8 }
{
st
r11
,
r0
; addi r11, r11, 8 }
{
ld
r0
,
r10
; addi r10, r10, 8 }
{
st
r11
,
r0
; addi r11, r11, 8 }
{
ld
r0
,
r10
; addi r10, r10, 8 }
{
st
r11
,
r0
}
{
flush
r11
; addi r11, r11, 8 }
cmpeq
r0
,
r33
,
r11
beqzt
r0
,
1
b
#ifdef RELOCATE_NEW_KERNEL_VERBOSE
moveli
r0
,
's'
jalr
r40
#endif
addi
r30
,
r30
,
8
j
.
Lloop
.
Lerr
:
moveli
r0
,
'e'
jalr
r40
moveli
r0
,
'r'
jalr
r40
moveli
r0
,
'r'
jalr
r40
moveli
r0
,
'\n'
jalr
r40
.
Lhalt
:
moveli
r41
,
hw2_last
(
hv_halt
)
shl16insli
r41
,
r41
,
hw1
(
hv_halt
)
shl16insli
r41
,
r41
,
hw0
(
hv_halt
)
jalr
r41
STD_ENDPROC
(
relocate_new_kernel
)
.
section
.
rodata
,
"a"
.
globl
relocate_new_kernel_size
relocate_new_kernel_size
:
.
long
.
Lend_relocate_new_kernel
-
relocate_new_kernel
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