blob: 0a321a448f4abda2faf9c727db50dec607fbb682 [file] [log] [blame]
Wolfgang Denk97caf672006-03-12 02:12:27 +01001/*
2 * U-boot - cache.c
3 *
Mike Frysinger79279c62008-02-19 00:50:58 -05004 * Copyright (c) 2005-2008 Analog Devices Inc.
Wolfgang Denk97caf672006-03-12 02:12:27 +01005 *
6 * (C) Copyright 2000-2004
7 * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
8 *
Mike Frysinger79279c62008-02-19 00:50:58 -05009 * Licensed under the GPL-2 or later.
Wolfgang Denk97caf672006-03-12 02:12:27 +010010 */
11
Aubrey.Li9da597f2007-03-09 13:38:44 +080012#include <common.h>
13#include <asm/blackfin.h>
Mike Frysingera047bc42008-08-07 15:21:47 -040014#include <asm/mach-common/bits/mpu.h>
Aubrey.Li9da597f2007-03-09 13:38:44 +080015
Mike Frysinger79279c62008-02-19 00:50:58 -050016void flush_cache(unsigned long addr, unsigned long size)
Wolfgang Denk97caf672006-03-12 02:12:27 +010017{
Mike Frysingereba7a042008-11-04 00:04:03 -050018 void *start_addr, *end_addr;
19 int istatus, dstatus;
20
Mike Frysinger79279c62008-02-19 00:50:58 -050021 /* no need to flush stuff in on chip memory (L1/L2/etc...) */
22 if (addr >= 0xE0000000)
Aubrey.Li9da597f2007-03-09 13:38:44 +080023 return;
24
Mike Frysingereba7a042008-11-04 00:04:03 -050025 start_addr = (void *)addr;
26 end_addr = (void *)(addr + size);
27 istatus = icache_status();
28 dstatus = dcache_status();
Aubrey.Li9da597f2007-03-09 13:38:44 +080029
Mike Frysingereba7a042008-11-04 00:04:03 -050030 if (istatus) {
31 if (dstatus)
32 blackfin_icache_dcache_flush_range(start_addr, end_addr);
33 else
34 blackfin_icache_flush_range(start_addr, end_addr);
35 } else if (dstatus)
36 blackfin_dcache_flush_range(start_addr, end_addr);
Wolfgang Denk97caf672006-03-12 02:12:27 +010037}
Mike Frysingera047bc42008-08-07 15:21:47 -040038
Mike Frysingera90f7902008-08-07 15:33:09 -040039#ifdef CONFIG_DCACHE_WB
40static void flushinv_all_dcache(void)
41{
42 u32 way, bank, subbank, set;
43 u32 status, addr;
44 u32 dmem_ctl = bfin_read_DMEM_CONTROL();
45
46 for (bank = 0; bank < 2; ++bank) {
47 if (!(dmem_ctl & (1 << (DMC1_P - bank))))
48 continue;
49
50 for (way = 0; way < 2; ++way)
51 for (subbank = 0; subbank < 4; ++subbank)
52 for (set = 0; set < 64; ++set) {
53
54 bfin_write_DTEST_COMMAND(
55 way << 26 |
56 bank << 23 |
57 subbank << 16 |
58 set << 5
59 );
60 CSYNC();
61 status = bfin_read_DTEST_DATA0();
62
63 /* only worry about valid/dirty entries */
64 if ((status & 0x3) != 0x3)
65 continue;
66
67 /* construct the address using the tag */
68 addr = (status & 0xFFFFC800) | (subbank << 12) | (set << 5);
69
70 /* flush it */
71 __asm__ __volatile__("FLUSHINV[%0];" : : "a"(addr));
72 }
73 }
74}
75#endif
76
Mike Frysingera047bc42008-08-07 15:21:47 -040077void icache_enable(void)
78{
79 bfin_write_IMEM_CONTROL(IMC | ENICPLB);
80 SSYNC();
81}
82
83void icache_disable(void)
84{
85 bfin_write_IMEM_CONTROL(0);
86 SSYNC();
87}
88
89int icache_status(void)
90{
Mike Frysinger24e6cf42008-08-07 18:40:13 -040091 return bfin_read_IMEM_CONTROL() & IMC;
Mike Frysingera047bc42008-08-07 15:21:47 -040092}
93
94void dcache_enable(void)
95{
96 bfin_write_DMEM_CONTROL(ACACHE_BCACHE | ENDCPLB | PORT_PREF0);
97 SSYNC();
98}
99
100void dcache_disable(void)
101{
Mike Frysingera90f7902008-08-07 15:33:09 -0400102#ifdef CONFIG_DCACHE_WB
103 bfin_write_DMEM_CONTROL(bfin_read_DMEM_CONTROL() & ~(ENDCPLB));
104 flushinv_all_dcache();
105#endif
Mike Frysingera047bc42008-08-07 15:21:47 -0400106 bfin_write_DMEM_CONTROL(0);
107 SSYNC();
108}
109
110int dcache_status(void)
111{
Mike Frysinger24e6cf42008-08-07 18:40:13 -0400112 return bfin_read_DMEM_CONTROL() & ACACHE_BCACHE;
Mike Frysingera047bc42008-08-07 15:21:47 -0400113}