OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [rc203soc/] [sw/] [uClinux/] [arch/] [armnommu/] [mm/] [proc-arm6,7.S] - Rev 1780

Go to most recent revision | Compare with Previous | Blame | View Log

/*
 * linux/arch/arm/mm/arm6.S: MMU functions for ARM6
 *
 * (C) 1997 Russell King
 *
 * These are the low level assembler for performing cache and TLB
 * functions on the ARM6 & ARM7.
 */

#include <asm/assembler.h>
#include "../lib/constants.h"
#define current current_set

/*
 * Function: arm6_7_flush_cache_all (void)
 *         : arm6_7_flush_cache_page (unsigned long address, int size, int flags)
 *
 * Params  : address    Area start address
 *         : size       size of area
 *         : flags      b0 = I cache as well
 *
 * Purpose : Flush all cache lines
 */
arm6_7_flush_cache:
                mov     r0, #0
#ifndef NO_MM
                mcr     p15, 0, r0, c7, c0, 0           @ flush cache
#endif
arm6_7_null:
                mov     pc, lr

/*
 * Function: arm6_7_flush_tlb_all (void)
 *
 * Purpose : flush all TLB entries in all caches
 */
arm6_7_flush_tlb_all:
                mov     r0, #0
#ifndef NO_MM
                mcr     p15, 0, r0, c5, c0, 0           @ flush TLB
#endif
                mov     pc, lr

/*
 * Function: arm6_7_flush_tlb_page (unsigned long address, int end, int flags)
 *
 * Params  : address    Area start address
 *         : end        Area end address
 *         : flags      b0 = I cache as well
 *
 * Purpose : flush a TLB entry
*/
arm6_7_flush_tlb_area:
#ifndef NO_MM
1:              mcr     p15, 0, r0, c6, c0, 0           @ flush TLB
                add     r0, r0, #4096
                cmp     r0, r1
                blt     1b
#endif
                mov     pc, lr

LC0:            .word   current


/*
 * Function: arm6_7_switch_to (struct task_struct *prev, struct task_struct *next)
 *
 * Params  : prev       Old task structure
 *         : next       New task structure for process to run
 *
 * Purpose : Perform a task switch, saving the old processes state, and restoring
 *           the new.
 *
 * Notes   : We don't fiddle with the FP registers here - we postpone this until
 *           the new task actually uses FP.  This way, we don't swap FP for tasks
 *           that do not require it.
 */
arm6_7_switch_to:
                stmfd   sp!, {r4 - r10, fp, lr}         @ Store most regs on stack
                mrs     ip, cpsr
                stmfd   sp!, {ip}                       @ Save cpsr_SVC
                str     sp, [r0, #TSS_SAVE]             @ Save sp_SVC
                ldr     r2, LC0
                str     r1, [r2]
                ldr     sp, [r1, #TSS_SAVE]             @ Get saved sp_SVC
#ifndef NO_MM 
                ldr     r0, [r1, #TSS_MEMMAP]           @ Page table pointer
                mov     r1, #0
                mcr     p15, 0, r1, c7, c0, 0           @ flush cache
                mcr     p15, 0, r0, c2, c0, 0           @ update page table ptr
                mcr     p15, 0, r1, c5, c0, 0           @ flush TLBs
#endif
                ldmfd   sp!, {ip}
                msr     spsr, ip                        @ Save tasks CPSR into SPSR for this return
                ldmfd   sp!, {r4 - r10, fp, pc}^                @ Load all regs saved previously

/*
 * Function: arm6_7_data_abort ()
 *
 * Params  : r0 = address of aborted instruction
 *
 * Purpose : obtain information about current aborted instruction
 *
 * Returns : r0 = address of abort
 *         : r1 = FSR
 *         : r2 != 0 if writing
 *         : sp = pointer to registers
 *
 * Notes   : data abort *** NEEDS WORK ***
 */

Lukabttxt:      .ascii  "Unknown data abort code %d [pc=%p, *pc=%p] LR=%p\0"
                .align

msg:            .ascii  "DA*%p=%p\n\0"
                .align

arm6_data_abort:
                ldr     r4, [r0]                        @ read instruction causing problem
                mov     r2, r4, lsr #19                 @ r2 b1 = L
                and     r1, r4, #15 << 24
                add     pc, pc, r1, lsr #22             @ Now branch to the relevent processing routine
                movs    pc, lr
                b       Ldata_unknown
                b       Ldata_unknown
                b       Ldata_unknown
                b       Ldata_unknown
                b       Ldata_earlyldrpost              @ ldr   rd, [rn], #m
                b       Ldata_simple                    @ ldr   rd, [rn, #m]    @ RegVal
                b       Ldata_earlyldrpost              @ ldr   rd, [rn], rm
                b       Ldata_simple                    @ ldr   rd, [rn, rm]
                b       Ldata_ldmstm                    @ ldm*a rn, <rlist>
                b       Ldata_ldmstm                    @ ldm*b rn, <rlist>
                b       Ldata_unknown
                b       Ldata_unknown
                b       Ldata_simple                    @ ldc   rd, [rn], #m    @ Same as ldr   rd, [rn], #m
                b       Ldata_simple                    @ ldc   rd, [rn, #m]
                b       Ldata_unknown
Ldata_unknown:  @ Part of jumptable
                ldr     r3, [sp, #15 * 4]               @ Get PC
                str     r3, [sp, #-4]!
                mov     r1, r1, lsr #2
                mov     r3, r4
                mov     r2, r0
                adr     r0, Lukabttxt
                bl      panic
Lstop:          b       Lstop

arm7_data_abort:
                ldr     r4, [r0]                        @ read instruction causing problem
                mov     r2, r4, lsr #19                 @ r2 b1 = L
                and     r1, r4, #15 << 24
                add     pc, pc, r1, lsr #22             @ Now branch to the relevent processing routine
                movs    pc, lr
                b       Ldata_unknown
                b       Ldata_unknown
                b       Ldata_unknown
                b       Ldata_unknown
                b       Ldata_lateldrpostconst          @ ldr   rd, [rn], #m
                b       Ldata_lateldrpreconst           @ ldr   rd, [rn, #m]    @ RegVal
                b       Ldata_lateldrpostreg            @ ldr   rd, [rn], rm
                b       Ldata_lateldrprereg             @ ldr   rd, [rn, rm]
                b       Ldata_ldmstm                    @ ldm*a rn, <rlist>
                b       Ldata_ldmstm                    @ ldm*b rn, <rlist>
                b       Ldata_unknown
                b       Ldata_unknown
                b       Ldata_simple                    @ ldc   rd, [rn], #m    @ Same as ldr   rd, [rn], #m
                b       Ldata_simple                    @ ldc   rd, [rn, #m]
                b       Ldata_unknown
                b       Ldata_unknown

Ldata_ldmstm:   tst     r4, #1 << 21                    @ check writeback bit
                beq     Ldata_simple

                mov     r7, #0x11
                orr     r7, r7, r7, lsl #8
                and     r0, r4, r7
                and     r1, r4, r7, lsl #1
                add     r0, r0, r1, lsr #1
                and     r1, r4, r7, lsl #2
                add     r0, r0, r1, lsr #2
                and     r1, r4, r7, lsl #3
                add     r0, r0, r1, lsr #3
                add     r0, r0, r0, lsr #8
                add     r0, r0, r0, lsr #4
                and     r7, r0, #15                     @ r7 = no. of registers to transfer.
                and     r5, r4, #15 << 16               @ Get Rn
                ldr     r0, [sp, r5, lsr #14]           @ Get register
                eor     r6, r4, r4, lsl #2
                tst     r6, #1 << 23                    @ Check inc/dec ^ writeback
                rsbeq   r7, r7, #0
                add     r7, r0, r7, lsl #2              @ Do correction (signed)
                str     r7, [sp, r5, lsr #14]           @ Put register

Ldata_simple:   and     r2, r2, #2                      @ check read/write bit
                mrc     p15, 0, r0, c6, c0, 0           @ get FAR
                mrc     p15, 0, r1, c5, c0, 0           @ get FSR
                and     r1, r1, #15
                mov     pc, lr

Ldata_earlyldrpost:
                tst     r2, #4
                and     r2, r2, #2                      @ check read/write bit
                orrne   r2, r2, #1                      @ T bit
                mrc     p15, 0, r0, c6, c0, 0           @ get FAR
                mrc     p15, 0, r1, c5, c0, 0           @ get FSR
                and     r1, r1, #15
                mov     pc, lr

Ldata_lateldrpostconst:
                movs    r1, r4, lsl #20                 @ Get offset
                beq     Ldata_earlyldrpost              @ if offset is zero, no effect
                and     r5, r4, #15 << 16               @ Get Rn
                ldr     r0, [sp, r5, lsr #14]
                tst     r4, #1 << 23                    @ U bit
                subne   r0, r0, r1, lsr #20
                addeq   r0, r0, r1, lsr #20
                str     r0, [sp, r5, lsr #14]           @ Put register
                b       Ldata_earlyldrpost

Ldata_lateldrpreconst:
                tst     r4, #1 << 21                    @ check writeback bit
                movnes  r1, r4, lsl #20                 @ Get offset
                beq     Ldata_simple
                and     r5, r4, #15 << 16               @ Get Rn
                ldr     r0, [sp, r5, lsr #14]
                tst     r4, #1 << 23                    @ U bit
                subne   r0, r0, r1, lsr #20
                addeq   r0, r0, r1, lsr #20
                str     r0, [sp, r5, lsr #14]           @ Put register
                b       Ldata_simple

Ldata_lateldrpostreg:
                and     r5, r4, #15
                ldr     r1, [sp, r5, lsl #2]            @ Get Rm
                mov     r3, r4, lsr #7
                ands    r3, r3, #31
                and     r6, r4, #0x70
                orreq   r6, r6, #8
                add     pc, pc, r6
                mov     r0, r0

                mov     r1, r1, lsl r3                  @ 0: LSL #!0
                b       1f
                b       1f                              @ 1: LSL #0
                mov     r0, r0
                b       1f                              @ 2: MUL?
                mov     r0, r0
                b       1f                              @ 3: MUL?
                mov     r0, r0
                mov     r1, r1, lsr r3                  @ 4: LSR #!0
                b       1f
                mov     r1, r1, lsr #32                 @ 5: LSR #32
                b       1f
                b       1f                              @ 6: MUL?
                mov     r0, r0
                b       1f                              @ 7: MUL?
                mov     r0, r0
                mov     r1, r1, asr r3                  @ 8: ASR #!0
                b       1f
                mov     r1, r1, asr #32                 @ 9: ASR #32
                b       1f
                b       1f                              @ A: MUL?
                mov     r0, r0
                b       1f                              @ B: MUL?
                mov     r0, r0
                mov     r1, r1, ror r3                  @ C: ROR #!0
                b       1f
                mov     r1, r1, rrx                     @ D: RRX
                b       1f
                mov     r0, r0                          @ E: MUL?
                mov     r0, r0
                mov     r0, r0                          @ F: MUL?


1:              and     r5, r4, #15 << 16               @ Get Rn
                ldr     r0, [sp, r5, lsr #14]
                tst     r4, #1 << 23                    @ U bit
                subne   r0, r0, r1
                addeq   r0, r0, r1
                str     r0, [sp, r5, lsr #14]           @ Put register
                b       Ldata_earlyldrpost

Ldata_lateldrprereg:
                tst     r4, #1 << 21                    @ check writeback bit
                beq     Ldata_simple
                and     r5, r4, #15
                ldr     r1, [sp, r5, lsl #2]            @ Get Rm
                mov     r3, r4, lsr #7
                ands    r3, r3, #31
                and     r6, r4, #0x70
                orreq   r6, r6, #8
                add     pc, pc, r6
                mov     r0, r0

                mov     r1, r1, lsl r3                  @ 0: LSL #!0
                b       1f
                b       1f                              @ 1: LSL #0
                mov     r0, r0
                b       1f                              @ 2: MUL?
                mov     r0, r0
                b       1f                              @ 3: MUL?
                mov     r0, r0
                mov     r1, r1, lsr r3                  @ 4: LSR #!0
                b       1f
                mov     r1, r1, lsr #32                 @ 5: LSR #32
                b       1f
                b       1f                              @ 6: MUL?
                mov     r0, r0
                b       1f                              @ 7: MUL?
                mov     r0, r0
                mov     r1, r1, asr r3                  @ 8: ASR #!0
                b       1f
                mov     r1, r1, asr #32                 @ 9: ASR #32
                b       1f
                b       1f                              @ A: MUL?
                mov     r0, r0
                b       1f                              @ B: MUL?
                mov     r0, r0
                mov     r1, r1, ror r3                  @ C: ROR #!0
                b       1f
                mov     r1, r1, rrx                     @ D: RRX
                b       1f
                mov     r0, r0                          @ E: MUL?
                mov     r0, r0
                mov     r0, r0                          @ F: MUL?


1:              and     r5, r4, #15 << 16               @ Get Rn
                ldr     r0, [sp, r5, lsr #14]
                tst     r4, #1 << 23                    @ U bit
                subne   r0, r0, r1
                addeq   r0, r0, r1
                str     r0, [sp, r5, lsr #14]           @ Put register
                b       Ldata_simple

/*
 * Function: arm6_7_check_bugs (void)
 *         : arm6_7_proc_init (void)
 *         : arm6_7_proc_fin (void)
 *
 * Notes   : This processor does not require these
 */
arm6_7_check_bugs:
                mrs     ip, cpsr
                bic     ip, ip, #F_BIT
                msr     cpsr, ip
arm6_7_proc_init:
arm6_7_proc_fin:
                mov     pc, lr

/*
 * Function: arm6_set_pmd ()
 *
 * Params  : r0 = Address to set
 *         : r1 = value to set
 *
 * Purpose : Set a PMD and flush it out of any WB cache
 */
arm6_set_pmd:   and     r2, r1, #3
                teq     r2, #2
                andeq   r2, r1, #8
                orreq   r1, r1, r2, lsl #1              @ Updatable = Cachable
                teq     r2, #1
                orreq   r1, r1, #16                     @ Updatable = 1 if Page table
                str     r1, [r0]
                mov     pc, lr

/*
 * Function: arm7_set_pmd ()
 *
 * Params  : r0 = Address to set
 *         : r1 = value to set
 *
 * Purpose : Set a PMD and flush it out of any WB cache
 */
arm7_set_pmd:
#ifndef NO_MM
        orr     r1, r1, #16                     @ Updatable bit is always set on ARM7
                str     r1, [r0]
#endif
                mov     pc, lr

/*
 * Function: arm6_7_reset
 *
 * Notes   : This sets up everything for a reset
 */
arm6_7_reset:   mrs     r1, cpsr
                orr     r1, r1, #F_BIT|I_BIT
                msr     cpsr, r1
                mov     r0, #0
#ifndef NO_MM
                mcr     p15, 0, r0, c7, c0, 0           @ flush cache
                mcr     p15, 0, r0, c5, c0, 0           @ flush TLB
#endif
                mov     r1, #F_BIT | I_BIT | 3
                mov     pc, lr

/*
 * Purpose : Function pointers used to access above functions - all calls
 *           come through these
 */
arm6_name:      .ascii  "arm6\0"
                .align

                .globl  arm6_processor_functions
arm6_processor_functions:
                .word   arm6_name                       @  0
                .word   arm6_7_switch_to                @  4
                .word   arm6_data_abort         @  8
                .word   arm6_7_check_bugs               @ 12
                .word   arm6_7_proc_init                @ 16
                .word   arm6_7_proc_fin         @ 20

                .word   arm6_7_flush_cache              @ 24
                .word   arm6_7_flush_cache              @ 28
                .word   arm6_7_flush_cache              @ 32
                .word   arm6_7_null                     @ 36
                .word   arm6_7_flush_cache              @ 40
                .word   arm6_7_flush_tlb_all            @ 44
                .word   arm6_7_flush_tlb_area           @ 48
                .word   arm6_set_pmd                    @ 52
                .word   arm6_7_reset                    @ 54

/*
 * Purpose : Function pointers used to access above functions - all calls
 *           come through these
 */
arm7_name:      .ascii  "arm7\0"
                .align

                .globl  arm7_processor_functions
arm7_processor_functions:
                .word   arm7_name                       @  0
                .word   arm6_7_switch_to                @  4
                .word   arm7_data_abort         @  8
                .word   arm6_7_check_bugs               @ 12
                .word   arm6_7_proc_init                @ 16
                .word   arm6_7_proc_fin         @ 20
                .word   arm6_7_flush_cache              @ 24
                .word   arm6_7_flush_cache              @ 28
                .word   arm6_7_flush_cache              @ 32
                .word   arm6_7_null                     @ 36
                .word   arm6_7_flush_cache              @ 40
                .word   arm6_7_flush_tlb_all            @ 44
                .word   arm6_7_flush_tlb_area           @ 48
                .word   arm7_set_pmd                    @ 52
                .word   arm6_7_reset                    @ 54

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.