1 /*This file is subject to the terms and conditions of the GNU General Public
4 * Blackfin BF533/2.6 support : LG Soft India
5 * Modification: Dec 07 2004
6 * 1. Correction in icheck_lock. Valid lock entries were
7 * geting victimized, for instruction cplb replacement.
8 * 2. Setup loop's are modified as now toolchain support's P Indexed
14 /* Usage: int _cplb_mgr(is_data_miss,int enable_cache)
15 * is_data_miss==2 => Mark as Dirty, write to the clean data page
16 * is_data_miss==1 => Replace a data CPLB.
17 * is_data_miss==0 => Replace an instruction CPLB.
20 * CPLB_RELOADED => Successfully updated CPLB table.
21 * CPLB_NO_UNLOCKED => All CPLBs are locked, so cannot be evicted.This indicates
22 * that the CPLBs in the configuration tablei are badly
23 * configured, as this should never occur.
24 * CPLB_NO_ADDR_MATCH => The address being accessed, that triggered the exception,
25 * is not covered by any of the CPLBs in the configuration
26 * table. The application isi presumably misbehaving.
27 * CPLB_PROT_VIOL => The address being accessed, that triggered thei exception,
28 * was not a first-write to a clean Write Back Data page,
29 * and so presumably is a genuine violation of the page's
30 * protection attributes. The application is misbehaving.
34 #include <asm-blackfin/linkage.h>
35 #include <asm-blackfin/blackfin.h>
36 #include <asm-blackfin/cplbtab.h>
37 #include <asm-blackfin/cplb.h>
47 IF CC JUMP dcplb_write;
50 IF !CC JUMP dcplb_miss_compare;
52 /* ICPLB Miss Exception. We need to choose one of the
53 * currently-installed CPLBs, and replace it with one
54 * from the configuration table.
57 P4.L = (ICPLB_FAULT_ADDR & 0xFFFF);
58 P4.H = (ICPLB_FAULT_ADDR >> 16);
61 P5.L = page_size_table;
62 P5.H = page_size_table;
64 P0.L = (ICPLB_DATA0 & 0xFFFF);
65 P0.H = (ICPLB_DATA0 >> 16);
66 R4 = [P4]; /* Get faulting address*/
67 R6 = 64; /* Advance past the fault address, which*/
68 R6 = R6 + R4; /* we'll use if we find a match*/
69 R3 = ((16 << 8) | 2); /* Extract mask, bits 16 and 17.*/
74 R1 = [P0-0x100]; /* Address for this CPLB */
76 R0 = [P0++]; /* Info for this CPLB*/
77 CC = BITTST(R0,0); /* Is the CPLB valid?*/
78 IF !CC JUMP nomatch; /* Skip it, if not.*/
79 CC = R4 < R1(IU); /* If fault address less than page start*/
80 IF CC JUMP nomatch; /* then skip this one.*/
81 R2 = EXTRACT(R0,R3.L) (Z); /* Get page size*/
83 P1 = P5 + (P1<<2); /* index into page-size table*/
84 R2 = [P1]; /* Get the page size*/
85 R1 = R1 + R2; /* and add to page start, to get page end*/
86 CC = R4 < R1(IU); /* and see whether fault addr is in page.*/
87 IF !CC R4 = R6; /* If so, advance the address and finish loop.*/
88 IF !CC JUMP isearch_done;
92 CC = BITTST(R5, 4); /* i.e CC = R5 >= 16*/
96 I0 = R4; /* Fault address we'll search for*/
99 P0.L = (ICPLB_DATA0 & 0xFFFF);
100 P0.H = (ICPLB_DATA0 >> 16);
102 /* The replacement procedure for ICPLBs */
104 P4.L = (IMEM_CONTROL & 0xFFFF);
105 P4.H = (IMEM_CONTROL >> 16);
108 R5 = [P4]; /* Control Register*/
109 BITCLR(R5,ENICPLB_P);
111 SSYNC; /* SSYNC required before writing to IMEM_CONTROL. */
117 R1 = -1; /* end point comparison */
118 R3 = 16; /* counter */
120 /* Search through CPLBs for first non-locked entry */
121 /* Overwrite it by moving everyone else up by 1 */
126 IF CC JUMP all_locked;
127 CC = BITTST(R0, 0); /* an invalid entry is good */
128 IF !CC JUMP ifound_victim;
129 CC = BITTST(R0,1); /* but a locked entry isn't */
130 IF CC JUMP icheck_lock;
133 #ifdef CONFIG_CPLB_INFO
137 P3.L = ipdt_swapcount_table;
138 P3.H = ipdt_swapcount_table;
141 R2 = [P2]; /* address from config table */
145 IF CC JUMP icount_done;
155 LSETUP(is_move,ie_move) LC0;
163 /* We've made space in the ICPLB table, so that ICPLB15
164 * is now free to be overwritten. Next, we have to determine
165 * which CPLB we need to install, from the configuration
166 * table. This is a matter of getting the start-of-page
167 * addresses and page-lengths from the config table, and
168 * determining whether the fault address falls within that
174 #ifdef CONFIG_CPLB_INFO
175 P3.L = ipdt_swapcount_table;
176 P3.H = ipdt_swapcount_table;
179 P0.L = page_size_table;
180 P0.H = page_size_table;
182 /* Retrieve our fault address (which may have been advanced
183 * because the faulting instruction crossed a page boundary).
188 /* An extraction pattern, to get the page-size bits from
189 * the CPLB data entry. Bits 16-17, so two bits at posn 16.
193 inext: R4 = [P2++]; /* address from config table */
194 R2 = [P2++]; /* data from config table */
195 #ifdef CONFIG_CPLB_INFO
199 CC = R4 == -1; /* End of config table*/
200 IF CC JUMP no_page_in_table;
202 /* See if failed address > start address */
206 /* extract page size (17:16)*/
207 R3 = EXTRACT(R2, R1.L) (Z);
209 /* add page size to addr to get range */
212 P5 = P0 + (P5 << 2); /* scaled, for int access*/
216 /* See if failed address < (start address + page size) */
220 /* We've found a CPLB in the config table that covers
221 * the faulting address, so install this CPLB into the
222 * last entry of the table.
225 P1.L = (ICPLB_DATA15 & 0xFFFF); /*ICPLB_DATA15*/
226 P1.H = (ICPLB_DATA15 >> 16);
229 #ifdef CONFIG_CPLB_INFO
235 /* P4 points to IMEM_CONTROL, and R5 contains its old
236 * value, after we disabled ICPLBS. Re-enable them.
239 BITSET(R5,ENICPLB_P);
241 SSYNC; /* SSYNC required before writing to IMEM_CONTROL. */
247 ( R7:0,P5:0 ) = [SP++];
253 ( R7:0,P5:0 ) = [SP++];
254 R0 = CPLB_NO_ADDR_MATCH;
257 ( R7:0,P5:0 ) = [SP++];
258 R0 = CPLB_NO_UNLOCKED;
261 ( R7:0,P5:0 ) = [SP++];
267 /* if a DCPLB is marked as write-back (CPLB_WT==0), and
268 * it is clean (CPLB_DIRTY==0), then a write to the
269 * CPLB's page triggers a protection violation. We have to
270 * mark the CPLB as dirty, to indicate that there are
271 * pending writes associated with the CPLB.
274 P4.L = (DCPLB_STATUS & 0xFFFF);
275 P4.H = (DCPLB_STATUS >> 16);
276 P3.L = (DCPLB_DATA0 & 0xFFFF);
277 P3.H = (DCPLB_DATA0 >> 16);
280 /* A protection violation can be caused by more than just writes
281 * to a clean WB page, so we have to ensure that:
283 * - to a clean WB page
284 * - and is allowed in the mode the access occurred.
287 CC = BITTST(R5, 16); /* ensure it was a write*/
288 IF !CC JUMP prot_violation;
290 /* to check the rest, we have to retrieve the DCPLB.*/
292 /* The low half of DCPLB_STATUS is a bit mask*/
294 R2 = R5.L (Z); /* indicating which CPLB triggered the event.*/
295 R3 = 30; /* so we can use this to determine the offset*/
297 R2 = R2.L (Z); /* into the DCPLB table.*/
301 R3 = [P3]; /* Retrieve the CPLB*/
303 /* Now we can check whether it's a clean WB page*/
305 CC = BITTST(R3, 14); /* 0==WB, 1==WT*/
306 IF CC JUMP prot_violation;
307 CC = BITTST(R3, 7); /* 0 == clean, 1 == dirty*/
308 IF CC JUMP prot_violation;
310 /* Check whether the write is allowed in the mode that was active.*/
312 R2 = 1<<3; /* checking write in user mode*/
313 CC = BITTST(R5, 17); /* 0==was user, 1==was super*/
315 R2 <<= R5; /* if was super, check write in super mode*/
318 IF CC JUMP prot_violation;
320 /* It's a genuine write-to-clean-page.*/
322 BITSET(R3, 7); /* mark as dirty*/
323 [P3] = R3; /* and write back.*/
325 ( R7:0,P5:0 ) = [SP++];
331 /* Data CPLB Miss event. We need to choose a CPLB to
332 * evict, and then locate a new CPLB to install from the
333 * config table, that covers the faulting address.
336 P1.L = (DCPLB_DATA15 & 0xFFFF);
337 P1.H = (DCPLB_DATA15 >> 16);
339 P4.L = (DCPLB_FAULT_ADDR & 0xFFFF);
340 P4.H = (DCPLB_FAULT_ADDR >> 16);
344 /* The replacement procedure for DCPLBs*/
346 R6 = R1; /* Save for later*/
348 /* Turn off CPLBs while we work.*/
349 P4.L = (DMEM_CONTROL & 0xFFFF);
350 P4.H = (DMEM_CONTROL >> 16);
352 BITCLR(R5,ENDCPLB_P);
354 SSYNC; /* SSYNC required before writing to DMEM_CONTROL. */
360 /* Start looking for a CPLB to evict. Our order of preference
361 * is: invalid CPLBs, clean CPLBs, dirty CPLBs. Locked CPLBs
365 I1.L = (DCPLB_DATA0 & 0xFFFF);
366 I1.H = (DCPLB_DATA0 >> 16);
369 I2.L = dcplb_preference;
370 I2.H = dcplb_preference;
371 LSETUP(sdsearch1, edsearch1) LC0 = P1;
373 R0 = [I2++]; /* Get the bits we're interested in*/
374 P0 = I1; /* Go back to start of table*/
375 LSETUP (sdsearch2, edsearch2) LC1 = P2;
377 R1 = [P0++]; /* Fetch each installed CPLB in turn*/
378 R2 = R1 & R0; /* and test for interesting bits.*/
379 CC = R2 == 0; /* If none are set, it'll do.*/
380 IF !CC JUMP skip_stack_check;
382 R2 = [P0 - 0x104]; /* R2 - PageStart */
383 P3.L = page_size_table; /* retrive end address */
384 P3.H = page_size_table; /* retrive end address */
385 R3 = 0x2; /* 0th - position, 2 bits -length */
386 nop; /*Anamoly 05000209*/
387 R7 = EXTRACT(R1,R3.l);
388 R7 = R7 << 2; /* Page size index offset */
391 R7 = [P3]; /* page size in 1K bytes */
393 R7 = R7 << 0xA; /* in bytes * 1024*/
394 R7 = R2 + R7; /* R7 - PageEnd */
395 R4 = SP; /* Test SP is in range */
397 CC = R7 < R4; /* if PageEnd < SP */
398 IF CC JUMP dfound_victim;
399 R3 = 0x284; /* stack length from start of trap till the point */
400 /* 20 stack locations for future modifications */
402 CC = R4 < R2; /* if SP + stacklen < PageStart */
403 IF CC JUMP dfound_victim;
409 /* If we got here, we didn't find a DCPLB we considered
410 * replacable, which means all of them were locked.
416 #ifdef CONFIG_CPLB_INFO
420 P3.L = dpdt_swapcount_table;
421 P3.H = dpdt_swapcount_table;
428 IF CC JUMP dicount_done;
438 /* Clean down the hardware loops*/
443 /* There's a suitable victim in [P0-4] (because we've
444 * advanced already). If it's a valid dirty write-back
445 * CPLB, we need to flush the pending writes first.
448 CC = BITTST(R1, 0); /* Is it valid?*/
449 IF !CC JUMP Ddoverwrite;/* nope.*/
450 CC = BITTST(R1, 7); /* Is it dirty?*/
451 IF !CC JUMP Ddoverwrite (BP); /* Nope.*/
452 CC = BITTST(R1, 14); /* Is it Write-Through?*/
453 IF CC JUMP Ddoverwrite; /* Yep*/
455 /* This is a dirty page, so we need to flush all writes
456 * that are pending on the page.
459 /* Retrieve the page start address*/
462 CALL dcplb_flush; /* R0==CPLB addr, R1==CPLB data*/
466 /* [P0-4] is a suitable victim CPLB, so we want to
467 * overwrite it by moving all the following CPLBs
468 * one space closer to the start.
471 R1.L = ((DCPLB_DATA15+4) & 0xFFFF); /*DCPLB_DATA15+4*/
472 R1.H = ((DCPLB_DATA15+4) >> 16);
475 /* If the victim happens to be in DCPLB15,
476 * we don't need to move anything.
484 LSETUP(ds_move, de_move) LC0=P1;
486 R0 = [P0++]; /* move data */
488 R0 = [P0-0x104] /* move address */
489 de_move: [P0-0x108] = R0;
491 /* We've now made space in DCPLB15 for the new CPLB to be
492 * installed. The next stage is to locate a CPLB in the
493 * config table that covers the faulting address.
497 R0 = I0; /* Our faulting address */
501 #ifdef CONFIG_CPLB_INFO
502 P3.L = dpdt_swapcount_table;
503 P3.H = dpdt_swapcount_table;
507 P1.L = page_size_table;
508 P1.H = page_size_table;
510 /* An extraction pattern, to retrieve bits 17:16.*/
513 dnext: R4 = [P2++]; /* address */
514 R2 = [P2++]; /* data */
515 #ifdef CONFIG_CPLB_INFO
520 IF CC JUMP no_page_in_table;
522 /* See if failed address > start address */
526 /* extract page size (17:16)*/
527 R3 = EXTRACT(R2, R1.L) (Z);
529 /* add page size to addr to get range */
536 /* See if failed address < (start address + page size) */
540 /* We've found the CPLB that should be installed, so
541 * write it into CPLB15, masking off any caching bits
545 P1.L = (DCPLB_DATA15 & 0xFFFF);
546 P1.H = (DCPLB_DATA15 >> 16);
548 /* If the DCPLB has cache bits set, but caching hasn't
549 * been enabled, then we want to mask off the cache-in-L1
550 * bit before installing. Moreover, if caching is off, we
551 * also want to ensure that the DCPLB has WT mode set, rather
552 * than WB, since WB pages still trigger first-write exceptions
553 * even when not caching is off, and the page isn't marked as
554 * cachable. Finally, we could mark the page as clean, not dirty,
555 * but we choose to leave that decision to the user; if the user
556 * chooses to have a CPLB pre-defined as dirty, then they always
557 * pay the cost of flushing during eviction, but don't pay the
558 * cost of first-write exceptions to mark the page as dirty.
561 #ifdef CONFIG_BLKFIN_WT
562 BITSET(R6, 14); /* Set WT*/
567 #ifdef CONFIG_CPLB_INFO
573 /* We've installed the CPLB, so re-enable CPLBs. P4
574 * points to DMEM_CONTROL, and R5 is the value we
575 * last wrote to it, when we were disabling CPLBs.
578 BITSET(R5,ENDCPLB_P);
585 ( R7:0,P5:0 ) = [SP++];
592 .byte4 0x00000400; /* 1K */
593 .byte4 0x00001000; /* 4K */
594 .byte4 0x00100000; /* 1M */
595 .byte4 0x00400000; /* 4M */
599 .byte4 0x00000001; /* valid bit */
600 .byte4 0x00000082; /* dirty+lock bits */
601 .byte4 0x00000002; /* lock bit */