1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright Altera Corporation (C) 2016. All rights reserved.
5 #include <linux/delay.h>
7 #include <linux/genalloc.h>
8 #include <linux/module.h>
9 #include <linux/of_address.h>
10 #include <linux/of_platform.h>
14 #define ALTR_OCRAM_CLEAR_ECC 0x00000018
15 #define ALTR_OCRAM_ECC_EN 0x00000019
17 void socfpga_init_ocram_ecc(void)
19 struct device_node
*np
;
20 void __iomem
*mapped_ocr_edac_addr
;
22 /* Find the OCRAM EDAC device tree node */
23 np
= of_find_compatible_node(NULL
, NULL
, "altr,socfpga-ocram-ecc");
25 pr_err("Unable to find socfpga-ocram-ecc\n");
29 mapped_ocr_edac_addr
= of_iomap(np
, 0);
31 if (!mapped_ocr_edac_addr
) {
32 pr_err("Unable to map OCRAM ecc regs.\n");
36 /* Clear any pending OCRAM ECC interrupts, then enable ECC */
37 writel(ALTR_OCRAM_CLEAR_ECC
, mapped_ocr_edac_addr
);
38 writel(ALTR_OCRAM_ECC_EN
, mapped_ocr_edac_addr
);
40 iounmap(mapped_ocr_edac_addr
);
43 /* Arria10 OCRAM Section */
44 #define ALTR_A10_ECC_CTRL_OFST 0x08
45 #define ALTR_A10_OCRAM_ECC_EN_CTL (BIT(1) | BIT(0))
46 #define ALTR_A10_ECC_INITA BIT(16)
48 #define ALTR_A10_ECC_INITSTAT_OFST 0x0C
49 #define ALTR_A10_ECC_INITCOMPLETEA BIT(0)
50 #define ALTR_A10_ECC_INITCOMPLETEB BIT(8)
52 #define ALTR_A10_ECC_ERRINTEN_OFST 0x10
53 #define ALTR_A10_ECC_SERRINTEN BIT(0)
55 #define ALTR_A10_ECC_INTSTAT_OFST 0x20
56 #define ALTR_A10_ECC_SERRPENA BIT(0)
57 #define ALTR_A10_ECC_DERRPENA BIT(8)
58 #define ALTR_A10_ECC_ERRPENA_MASK (ALTR_A10_ECC_SERRPENA | \
59 ALTR_A10_ECC_DERRPENA)
60 /* ECC Manager Defines */
61 #define A10_SYSMGR_ECC_INTMASK_SET_OFST 0x94
62 #define A10_SYSMGR_ECC_INTMASK_CLR_OFST 0x98
63 #define A10_SYSMGR_ECC_INTMASK_OCRAM BIT(1)
65 #define ALTR_A10_ECC_INIT_WATCHDOG_10US 10000
67 static inline void ecc_set_bits(u32 bit_mask
, void __iomem
*ioaddr
)
69 u32 value
= readl(ioaddr
);
72 writel(value
, ioaddr
);
75 static inline void ecc_clear_bits(u32 bit_mask
, void __iomem
*ioaddr
)
77 u32 value
= readl(ioaddr
);
80 writel(value
, ioaddr
);
83 static inline int ecc_test_bits(u32 bit_mask
, void __iomem
*ioaddr
)
85 u32 value
= readl(ioaddr
);
87 return (value
& bit_mask
) ? 1 : 0;
91 * This function uses the memory initialization block in the Arria10 ECC
92 * controller to initialize/clear the entire memory data and ECC data.
94 static int altr_init_memory_port(void __iomem
*ioaddr
)
96 int limit
= ALTR_A10_ECC_INIT_WATCHDOG_10US
;
98 ecc_set_bits(ALTR_A10_ECC_INITA
, (ioaddr
+ ALTR_A10_ECC_CTRL_OFST
));
100 if (ecc_test_bits(ALTR_A10_ECC_INITCOMPLETEA
,
101 (ioaddr
+ ALTR_A10_ECC_INITSTAT_OFST
)))
108 /* Clear any pending ECC interrupts */
109 writel(ALTR_A10_ECC_ERRPENA_MASK
,
110 (ioaddr
+ ALTR_A10_ECC_INTSTAT_OFST
));
115 void socfpga_init_arria10_ocram_ecc(void)
117 struct device_node
*np
;
119 void __iomem
*ecc_block_base
;
121 if (!sys_manager_base_addr
) {
122 pr_err("SOCFPGA: sys-mgr is not initialized\n");
126 /* Find the OCRAM EDAC device tree node */
127 np
= of_find_compatible_node(NULL
, NULL
, "altr,socfpga-a10-ocram-ecc");
129 pr_err("Unable to find socfpga-a10-ocram-ecc\n");
133 /* Map the ECC Block */
134 ecc_block_base
= of_iomap(np
, 0);
136 if (!ecc_block_base
) {
137 pr_err("Unable to map OCRAM ECC block\n");
142 writel(ALTR_A10_OCRAM_ECC_EN_CTL
,
143 sys_manager_base_addr
+ A10_SYSMGR_ECC_INTMASK_SET_OFST
);
144 ecc_clear_bits(ALTR_A10_ECC_SERRINTEN
,
145 (ecc_block_base
+ ALTR_A10_ECC_ERRINTEN_OFST
));
146 ecc_clear_bits(ALTR_A10_OCRAM_ECC_EN_CTL
,
147 (ecc_block_base
+ ALTR_A10_ECC_CTRL_OFST
));
149 /* Ensure all writes complete */
152 /* Use HW initialization block to initialize memory for ECC */
153 ret
= altr_init_memory_port(ecc_block_base
);
155 pr_err("ECC: cannot init OCRAM PORTA memory\n");
160 ecc_set_bits(ALTR_A10_OCRAM_ECC_EN_CTL
,
161 (ecc_block_base
+ ALTR_A10_ECC_CTRL_OFST
));
162 ecc_set_bits(ALTR_A10_ECC_SERRINTEN
,
163 (ecc_block_base
+ ALTR_A10_ECC_ERRINTEN_OFST
));
164 writel(ALTR_A10_OCRAM_ECC_EN_CTL
,
165 sys_manager_base_addr
+ A10_SYSMGR_ECC_INTMASK_CLR_OFST
);
167 /* Ensure all writes complete */
170 iounmap(ecc_block_base
);