| /* |
| * Based on: arch/blackfin/kernel/cplb-mpu/cplbmgr.c |
| * Author: Michael McTernan <mmcternan@airvana.com> |
| * |
| * Description: CPLB miss handler. |
| * |
| * Modified: |
| * Copyright 2008 Airvana Inc. |
| * Copyright 2008-2009 Analog Devices Inc. |
| * |
| * Licensed under the GPL-2 or later |
| */ |
| |
| #include <linux/kernel.h> |
| #include <asm/blackfin.h> |
| #include <asm/cplbinit.h> |
| #include <asm/cplb.h> |
| #include <asm/mmu_context.h> |
| #include <asm/traps.h> |
| |
| /* |
| * WARNING |
| * |
| * This file is compiled with certain -ffixed-reg options. We have to |
| * make sure not to call any functions here that could clobber these |
| * registers. |
| */ |
| |
| int nr_dcplb_miss[NR_CPUS], nr_icplb_miss[NR_CPUS]; |
| int nr_dcplb_supv_miss[NR_CPUS], nr_icplb_supv_miss[NR_CPUS]; |
| int nr_cplb_flush[NR_CPUS], nr_dcplb_prot[NR_CPUS]; |
| |
| #ifdef CONFIG_EXCPT_IRQ_SYSC_L1 |
| #define MGR_ATTR __attribute__((l1_text)) |
| #else |
| #define MGR_ATTR |
| #endif |
| |
| static inline void write_dcplb_data(int cpu, int idx, unsigned long data, |
| unsigned long addr) |
| { |
| _disable_dcplb(); |
| bfin_write32(DCPLB_DATA0 + idx * 4, data); |
| bfin_write32(DCPLB_ADDR0 + idx * 4, addr); |
| _enable_dcplb(); |
| |
| #ifdef CONFIG_CPLB_INFO |
| dcplb_tbl[cpu][idx].addr = addr; |
| dcplb_tbl[cpu][idx].data = data; |
| #endif |
| } |
| |
| static inline void write_icplb_data(int cpu, int idx, unsigned long data, |
| unsigned long addr) |
| { |
| _disable_icplb(); |
| bfin_write32(ICPLB_DATA0 + idx * 4, data); |
| bfin_write32(ICPLB_ADDR0 + idx * 4, addr); |
| _enable_icplb(); |
| |
| #ifdef CONFIG_CPLB_INFO |
| icplb_tbl[cpu][idx].addr = addr; |
| icplb_tbl[cpu][idx].data = data; |
| #endif |
| } |
| |
| /* Counters to implement round-robin replacement. */ |
| static int icplb_rr_index[NR_CPUS] PDT_ATTR; |
| static int dcplb_rr_index[NR_CPUS] PDT_ATTR; |
| |
| /* |
| * Find an ICPLB entry to be evicted and return its index. |
| */ |
| static int evict_one_icplb(int cpu) |
| { |
| int i = first_switched_icplb + icplb_rr_index[cpu]; |
| if (i >= MAX_CPLBS) { |
| i -= MAX_CPLBS - first_switched_icplb; |
| icplb_rr_index[cpu] -= MAX_CPLBS - first_switched_icplb; |
| } |
| icplb_rr_index[cpu]++; |
| return i; |
| } |
| |
| static int evict_one_dcplb(int cpu) |
| { |
| int i = first_switched_dcplb + dcplb_rr_index[cpu]; |
| if (i >= MAX_CPLBS) { |
| i -= MAX_CPLBS - first_switched_dcplb; |
| dcplb_rr_index[cpu] -= MAX_CPLBS - first_switched_dcplb; |
| } |
| dcplb_rr_index[cpu]++; |
| return i; |
| } |
| |
| MGR_ATTR static int icplb_miss(int cpu) |
| { |
| unsigned long addr = bfin_read_ICPLB_FAULT_ADDR(); |
| int status = bfin_read_ICPLB_STATUS(); |
| int idx; |
| unsigned long i_data, base, addr1, eaddr; |
| |
| nr_icplb_miss[cpu]++; |
| if (unlikely(status & FAULT_USERSUPV)) |
| nr_icplb_supv_miss[cpu]++; |
| |
| base = 0; |
| idx = 0; |
| do { |
| eaddr = icplb_bounds[idx].eaddr; |
| if (addr < eaddr) |
| break; |
| base = eaddr; |
| } while (++idx < icplb_nr_bounds); |
| |
| if (unlikely(idx == icplb_nr_bounds)) |
| return CPLB_NO_ADDR_MATCH; |
| |
| i_data = icplb_bounds[idx].data; |
| if (unlikely(i_data == 0)) |
| return CPLB_NO_ADDR_MATCH; |
| |
| addr1 = addr & ~(SIZE_4M - 1); |
| addr &= ~(SIZE_1M - 1); |
| i_data |= PAGE_SIZE_1MB; |
| if (addr1 >= base && (addr1 + SIZE_4M) <= eaddr) { |
| /* |
| * This works because |
| * (PAGE_SIZE_4MB & PAGE_SIZE_1MB) == PAGE_SIZE_1MB. |
| */ |
| i_data |= PAGE_SIZE_4MB; |
| addr = addr1; |
| } |
| |
| /* Pick entry to evict */ |
| idx = evict_one_icplb(cpu); |
| |
| write_icplb_data(cpu, idx, i_data, addr); |
| |
| return CPLB_RELOADED; |
| } |
| |
| MGR_ATTR static int dcplb_miss(int cpu) |
| { |
| unsigned long addr = bfin_read_DCPLB_FAULT_ADDR(); |
| int status = bfin_read_DCPLB_STATUS(); |
| int idx; |
| unsigned long d_data, base, addr1, eaddr, cplb_pagesize, cplb_pageflags; |
| |
| nr_dcplb_miss[cpu]++; |
| if (unlikely(status & FAULT_USERSUPV)) |
| nr_dcplb_supv_miss[cpu]++; |
| |
| base = 0; |
| idx = 0; |
| do { |
| eaddr = dcplb_bounds[idx].eaddr; |
| if (addr < eaddr) |
| break; |
| base = eaddr; |
| } while (++idx < dcplb_nr_bounds); |
| |
| if (unlikely(idx == dcplb_nr_bounds)) |
| return CPLB_NO_ADDR_MATCH; |
| |
| d_data = dcplb_bounds[idx].data; |
| if (unlikely(d_data == 0)) |
| return CPLB_NO_ADDR_MATCH; |
| |
| addr &= ~(SIZE_1M - 1); |
| d_data |= PAGE_SIZE_1MB; |
| |
| /* BF60x support large than 4M CPLB page size */ |
| #ifdef PAGE_SIZE_16MB |
| cplb_pageflags = PAGE_SIZE_16MB; |
| cplb_pagesize = SIZE_16M; |
| #else |
| cplb_pageflags = PAGE_SIZE_4MB; |
| cplb_pagesize = SIZE_4M; |
| #endif |
| |
| find_pagesize: |
| addr1 = addr & ~(cplb_pagesize - 1); |
| if (addr1 >= base && (addr1 + cplb_pagesize) <= eaddr) { |
| /* |
| * This works because |
| * (PAGE_SIZE_4MB & PAGE_SIZE_1MB) == PAGE_SIZE_1MB. |
| */ |
| d_data |= cplb_pageflags; |
| addr = addr1; |
| goto found_pagesize; |
| } else { |
| if (cplb_pagesize > SIZE_4M) { |
| cplb_pageflags = PAGE_SIZE_4MB; |
| cplb_pagesize = SIZE_4M; |
| goto find_pagesize; |
| } |
| } |
| |
| found_pagesize: |
| #ifdef CONFIG_BF60x |
| if ((addr >= ASYNC_BANK0_BASE) |
| && (addr < ASYNC_BANK3_BASE + ASYNC_BANK3_SIZE)) |
| d_data |= PAGE_SIZE_64MB; |
| #endif |
| |
| /* Pick entry to evict */ |
| idx = evict_one_dcplb(cpu); |
| |
| write_dcplb_data(cpu, idx, d_data, addr); |
| |
| return CPLB_RELOADED; |
| } |
| |
| MGR_ATTR int cplb_hdr(int seqstat, struct pt_regs *regs) |
| { |
| int cause = seqstat & 0x3f; |
| unsigned int cpu = raw_smp_processor_id(); |
| switch (cause) { |
| case VEC_CPLB_I_M: |
| return icplb_miss(cpu); |
| case VEC_CPLB_M: |
| return dcplb_miss(cpu); |
| default: |
| return CPLB_UNKNOWN_ERR; |
| } |
| } |