blob: dce69abdfd1d4f2d615997f59cad475ba8df5fa7 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +02002/*
3 * (C) Copyright 2007-2008
Stelian Popc9e798d2011-11-01 00:00:39 +01004 * Stelian Pop <stelian@popies.net>
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +02005 * Lead Tech Design <www.leadtechdesign.com>
6 *
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +01007 * (C) Copyright 2009-2015
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +02008 * Daniel Gorsulowski <daniel.gorsulowski@esd.eu>
9 * esd electronic system design gmbh <www.esd.eu>
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020010 */
11
Tom Rini03de3052024-05-20 13:35:03 -060012#include <config.h>
Simon Glass3a7d5572019-08-01 09:46:42 -060013#include <env.h>
Simon Glass9b4a2052019-12-28 10:45:05 -070014#include <init.h>
Simon Glass90526e92020-05-10 11:39:56 -060015#include <net.h>
Simon Glassb03e0512019-11-14 12:57:24 -070016#include <serial.h>
Simon Glass2189d5f2019-11-14 12:57:20 -070017#include <vsprintf.h>
Simon Glass401d1c42020-10-30 21:38:53 -060018#include <asm/global_data.h>
Matthias Fuchs0cb77bf2011-07-19 01:56:06 +000019#include <asm/io.h>
Andreas Bießmannac45bb12013-11-29 12:13:45 +010020#include <asm/gpio.h>
Simon Glassc62db352017-05-31 19:47:48 -060021#include <asm/mach-types.h>
Simon Glass5d982852017-05-17 08:23:00 -060022#include <asm/setup.h>
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020023#include <asm/arch/at91sam9_smc.h>
24#include <asm/arch/at91_common.h>
25#include <asm/arch/at91_pmc.h>
26#include <asm/arch/at91_rstc.h>
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +020027#include <asm/arch/at91_matrix.h>
28#include <asm/arch/at91_pio.h>
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020029#include <asm/arch/clk.h>
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020030#include <netdev.h>
31
32DECLARE_GLOBAL_DATA_PTR;
33
34/*
35 * Miscelaneous platform dependent initialisations
36 */
37
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +010038#ifdef CONFIG_REVISION_TAG
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020039static int hw_rev = -1; /* hardware revision */
40
41int get_hw_rev(void)
42{
43 if (hw_rev >= 0)
44 return hw_rev;
45
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +020046 hw_rev = at91_get_pio_value(AT91_PIO_PORTB, 19);
47 hw_rev |= at91_get_pio_value(AT91_PIO_PORTB, 20) << 1;
48 hw_rev |= at91_get_pio_value(AT91_PIO_PORTB, 21) << 2;
49 hw_rev |= at91_get_pio_value(AT91_PIO_PORTB, 22) << 3;
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020050
51 if (hw_rev == 15)
52 hw_rev = 0;
53
54 return hw_rev;
55}
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +010056#endif /* CONFIG_REVISION_TAG */
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020057
58#ifdef CONFIG_CMD_NAND
59static void meesc_nand_hw_init(void)
60{
61 unsigned long csa;
Matthias Fuchs0cb77bf2011-07-19 01:56:06 +000062 at91_smc_t *smc = (at91_smc_t *) ATMEL_BASE_SMC0;
63 at91_matrix_t *matrix = (at91_matrix_t *) ATMEL_BASE_MATRIX;
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020064
65 /* Enable CS3 */
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +020066 csa = readl(&matrix->csa[0]) | AT91_MATRIX_CSA_EBI_CS3A;
67 writel(csa, &matrix->csa[0]);
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020068
69 /* Configure SMC CS3 for NAND/SmartMedia */
Daniel Gorsulowskidd802642012-01-25 03:19:49 +000070 writel(AT91_SMC_SETUP_NWE(1) | AT91_SMC_SETUP_NCS_WR(1) |
71 AT91_SMC_SETUP_NRD(2) | AT91_SMC_SETUP_NCS_RD(2),
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +020072 &smc->cs[3].setup);
73
74 writel(AT91_SMC_PULSE_NWE(3) | AT91_SMC_PULSE_NCS_WR(3) |
75 AT91_SMC_PULSE_NRD(3) | AT91_SMC_PULSE_NCS_RD(3),
76 &smc->cs[3].pulse);
77
Daniel Gorsulowskidd802642012-01-25 03:19:49 +000078 writel(AT91_SMC_CYCLE_NWE(6) | AT91_SMC_CYCLE_NRD(6),
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +020079 &smc->cs[3].cycle);
80 writel(AT91_SMC_MODE_RM_NRD | AT91_SMC_MODE_WM_NWE |
81 AT91_SMC_MODE_EXNW_DISABLE |
82 AT91_SMC_MODE_DBW_8 |
Daniel Gorsulowskidd802642012-01-25 03:19:49 +000083 AT91_SMC_MODE_TDF_CYCLE(12),
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +020084 &smc->cs[3].mode);
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020085
86 /* Configure RDY/BSY */
Tom Rini4e590942022-11-12 17:36:51 -050087 gpio_direction_input(CFG_SYS_NAND_READY_PIN);
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020088
89 /* Enable NandFlash */
Tom Rini4e590942022-11-12 17:36:51 -050090 gpio_direction_output(CFG_SYS_NAND_ENABLE_PIN, 1);
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020091}
92#endif /* CONFIG_CMD_NAND */
93
94#ifdef CONFIG_MACB
95static void meesc_macb_hw_init(void)
96{
Wenyou Yang70341e22016-02-03 10:16:50 +080097 at91_periph_clk_enable(ATMEL_ID_EMAC);
98
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +020099 at91_macb_hw_init();
100}
101#endif
102
103/*
104 * Static memory controller initialization to enable Beckhoff ET1100 EtherCAT
105 * controller debugging
106 * The ET1100 is located at physical address 0x70000000
107 * Its process memory is located at physical address 0x70001000
108 */
109static void meesc_ethercat_hw_init(void)
110{
Matthias Fuchs0cb77bf2011-07-19 01:56:06 +0000111 at91_smc_t *smc1 = (at91_smc_t *) ATMEL_BASE_SMC1;
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +0200112
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200113 /* Configure SMC EBI1_CS0 for EtherCAT */
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +0200114 writel(AT91_SMC_SETUP_NWE(0) | AT91_SMC_SETUP_NCS_WR(0) |
115 AT91_SMC_SETUP_NRD(0) | AT91_SMC_SETUP_NCS_RD(0),
116 &smc1->cs[0].setup);
117 writel(AT91_SMC_PULSE_NWE(4) | AT91_SMC_PULSE_NCS_WR(9) |
118 AT91_SMC_PULSE_NRD(5) | AT91_SMC_PULSE_NCS_RD(9),
119 &smc1->cs[0].pulse);
120 writel(AT91_SMC_CYCLE_NWE(10) | AT91_SMC_CYCLE_NRD(6),
121 &smc1->cs[0].cycle);
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200122 /*
123 * Configure behavior at external wait signal, byte-select mode, 16 bit
124 * data bus width, none data float wait states and TDF optimization
125 */
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +0200126 writel(AT91_SMC_MODE_RM_NRD | AT91_SMC_MODE_EXNW_READY |
127 AT91_SMC_MODE_DBW_16 | AT91_SMC_MODE_TDF_CYCLE(0) |
128 AT91_SMC_MODE_TDF, &smc1->cs[0].mode);
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200129
130 /* Configure RDY/BSY */
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +0200131 at91_set_b_periph(AT91_PIO_PORTE, 20, 0); /* EBI1_NWAIT */
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200132}
133
134int dram_init(void)
135{
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +0100136 /* dram_init must store complete ramsize in gd->ram_size */
137 gd->ram_size = get_ram_size((void *)PHYS_SDRAM,
138 PHYS_SDRAM_SIZE);
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200139 return 0;
140}
141
Simon Glass76b00ac2017-03-31 08:40:32 -0600142int dram_init_banksize(void)
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +0100143{
144 gd->bd->bi_dram[0].start = PHYS_SDRAM;
145 gd->bd->bi_dram[0].size = PHYS_SDRAM_SIZE;
Simon Glass76b00ac2017-03-31 08:40:32 -0600146
147 return 0;
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +0100148}
149
Masahiro Yamadab75d8dc2020-06-26 15:13:33 +0900150int board_eth_init(struct bd_info *bis)
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200151{
152 int rc = 0;
153#ifdef CONFIG_MACB
Matthias Fuchs0cb77bf2011-07-19 01:56:06 +0000154 rc = macb_eth_initialize(0, (void *)ATMEL_BASE_EMAC, 0x00);
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200155#endif
156 return rc;
157}
158
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +0100159#ifdef CONFIG_DISPLAY_BOARDINFO
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200160int checkboard(void)
161{
162 char str[32];
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200163 u_char hw_type; /* hardware type */
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200164
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200165 /* read the "Type" register of the ET1100 controller */
Tom Rinib9abcb82022-12-04 10:03:48 -0500166 hw_type = readb(CFG_ET1100_BASE);
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200167
168 switch (hw_type) {
169 case 0x11:
170 case 0x3F:
171 /* ET1100 present, arch number of MEESC-Board */
172 gd->bd->bi_arch_number = MACH_TYPE_MEESC;
173 puts("Board: CAN-EtherCAT Gateway");
174 break;
175 case 0xFF:
176 /* no ET1100 present, arch number of EtherCAN/2-Board */
177 gd->bd->bi_arch_number = MACH_TYPE_ETHERCAN2;
178 puts("Board: EtherCAN/2 Gateway");
179 /* switch on LED1D */
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +0200180 at91_set_pio_output(AT91_PIO_PORTB, 12, 1);
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200181 break;
182 default:
183 /* assume, no ET1100 present, arch number of EtherCAN/2-Board */
184 gd->bd->bi_arch_number = MACH_TYPE_ETHERCAN2;
185 printf("ERROR! Read invalid hw_type: %02X\n", hw_type);
186 puts("Board: EtherCAN/2 Gateway");
187 break;
188 }
Simon Glass00caae62017-08-03 12:22:12 -0600189 if (env_get_f("serial#", str, sizeof(str)) > 0) {
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200190 puts(", serial# ");
191 puts(str);
192 }
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +0100193#ifdef CONFIG_REVISION_TAG
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200194 printf("\nHardware-revision: 1.%d\n", get_hw_rev());
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +0100195#endif
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200196 printf("Mach-type: %lu\n", gd->bd->bi_arch_number);
197 return 0;
198}
Daniel Gorsulowski83bf0052015-11-02 07:59:49 +0100199#endif /* CONFIG_DISPLAY_BOARDINFO */
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200200
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200201#ifdef CONFIG_SERIAL_TAG
202void get_board_serial(struct tag_serialnr *serialnr)
203{
204 char *str;
205
Simon Glass00caae62017-08-03 12:22:12 -0600206 char *serial = env_get("serial#");
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200207 if (serial) {
208 str = strchr(serial, '_');
209 if (str && (strlen(str) >= 4)) {
210 serialnr->high = (*(str + 1) << 8) | *(str + 2);
Simon Glass7e5f4602021-07-24 09:03:29 -0600211 serialnr->low = hextoul(str + 3, NULL);
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200212 }
213 } else {
214 serialnr->high = 0;
215 serialnr->low = 0;
216 }
217}
218#endif
219
220#ifdef CONFIG_REVISION_TAG
221u32 get_board_rev(void)
222{
223 return hw_rev | 0x100;
224}
225#endif
226
Daniel Gorsulowskia3f38972010-01-20 08:00:11 +0100227#ifdef CONFIG_MISC_INIT_R
228int misc_init_r(void)
229{
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +0200230 char *str;
231 char buf[32];
Matthias Fuchs0cb77bf2011-07-19 01:56:06 +0000232 at91_pmc_t *pmc = (at91_pmc_t *) ATMEL_BASE_PMC;
Daniel Gorsulowskia3f38972010-01-20 08:00:11 +0100233
234 /*
235 * Normally the processor clock has a divisor of 2.
236 * In some cases this this needs to be set to 4.
237 * Check the user has set environment mdiv to 4 to change the divisor.
238 */
Simon Glass00caae62017-08-03 12:22:12 -0600239 str = env_get("mdiv");
240 if (str && (strcmp(str, "4") == 0)) {
Daniel Gorsulowskid4562e02010-08-09 11:17:14 +0200241 writel((readl(&pmc->mckr) & ~AT91_PMC_MDIV) |
242 AT91SAM9_PMC_MDIV_4, &pmc->mckr);
Tom Rini65cc0e22022-11-16 13:10:41 -0500243 at91_clock_init(CFG_SYS_AT91_MAIN_CLOCK);
Daniel Gorsulowskia3f38972010-01-20 08:00:11 +0100244 serial_setbrg();
245 /* Notify the user that the clock is not default */
246 printf("Setting master clock to %s MHz\n",
247 strmhz(buf, get_mck_clk_rate()));
248 }
249
250 return 0;
251}
252#endif /* CONFIG_MISC_INIT_R */
253
Matthias Fuchs0cb77bf2011-07-19 01:56:06 +0000254int board_early_init_f(void)
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200255{
Wenyou Yang70341e22016-02-03 10:16:50 +0800256 at91_periph_clk_enable(ATMEL_ID_UHP);
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200257
Matthias Fuchs0cb77bf2011-07-19 01:56:06 +0000258 return 0;
259}
260
261int board_init(void)
262{
Daniel Gorsulowskia3802792009-09-29 08:03:12 +0200263 /* initialize ET1100 Controller */
264 meesc_ethercat_hw_init();
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200265
266 /* adress of boot parameters */
Tom Riniaa6e94d2022-11-16 13:10:37 -0500267 gd->bd->bi_boot_params = CFG_SYS_SDRAM_BASE + 0x100;
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200268
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200269#ifdef CONFIG_CMD_NAND
270 meesc_nand_hw_init();
271#endif
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200272#ifdef CONFIG_MACB
273 meesc_macb_hw_init();
274#endif
275#ifdef CONFIG_AT91_CAN
276 at91_can_hw_init();
277#endif
Daniel Gorsulowski64037fb2010-08-09 11:17:15 +0200278#ifdef CONFIG_USB_OHCI_NEW
279 at91_uhp_hw_init();
280#endif
Daniel Gorsulowski33b1d3f2009-06-30 21:03:37 +0200281 return 0;
282}