Skip to content

Commit e328c2b

Browse files
pdgendtcarlescufi
authored andcommitted
drivers: memc: Introduce Atmel SAM SMC/EBI driver
Add a driver to support external memory connected to the SMC port for Atmel SAM devices. Signed-off-by: Pieter De Gendt <pieter.degendt@basalte.be>
1 parent 7ef6433 commit e328c2b

File tree

5 files changed

+270
-0
lines changed

5 files changed

+270
-0
lines changed

drivers/memc/CMakeLists.txt

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@ zephyr_library_sources_ifdef(CONFIG_MEMC_STM32_NOR_PSRAM memc_stm32_nor_psram.c)
1010
zephyr_library_sources_ifdef(CONFIG_MEMC_MCUX_FLEXSPI memc_mcux_flexspi.c)
1111
zephyr_library_sources_ifdef(CONFIG_MEMC_MCUX_FLEXSPI_HYPERRAM memc_mcux_flexspi_hyperram.c)
1212

13+
zephyr_library_sources_ifdef(CONFIG_MEMC_SAM_SMC memc_sam_smc.c)
14+
1315
if(CONFIG_FLASH_MCUX_FLEXSPI_XIP)
1416
zephyr_code_relocate(memc_mcux_flexspi.c ${CONFIG_FLASH_MCUX_FLEXSPI_XIP_MEM}_TEXT)
1517
endif()

drivers/memc/Kconfig

+2
Original file line numberDiff line numberDiff line change
@@ -25,4 +25,6 @@ source "drivers/memc/Kconfig.stm32"
2525

2626
source "drivers/memc/Kconfig.mcux"
2727

28+
source "drivers/memc/Kconfig.sam"
29+
2830
endif

drivers/memc/Kconfig.sam

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
# Copyright (c) 2022 Basalte bv
2+
# SPDX-License-Identifier: Apache-2.0
3+
4+
config MEMC_SAM_SMC
5+
bool "Atmel Static Memory Controller (SMC)"
6+
default y
7+
depends on DT_HAS_ATMEL_SAM_SMC_ENABLED
8+
help
9+
Enable Atmel Static Memory Controller.

drivers/memc/memc_sam_smc.c

+105
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,105 @@
1+
/*
2+
* Copyright (c) 2022 Basalte bv
3+
*
4+
* SPDX-License-Identifier: Apache-2.0
5+
*/
6+
7+
#define DT_DRV_COMPAT atmel_sam_smc
8+
9+
#include <zephyr/device.h>
10+
#include <zephyr/drivers/pinctrl.h>
11+
#include <soc.h>
12+
13+
#include <zephyr/logging/log.h>
14+
LOG_MODULE_REGISTER(memc_sam, CONFIG_MEMC_LOG_LEVEL);
15+
16+
struct memc_smc_bank_config {
17+
uint32_t cs;
18+
uint32_t mode;
19+
uint32_t setup_timing;
20+
uint32_t pulse_timing;
21+
uint32_t cycle_timing;
22+
};
23+
24+
struct memc_smc_config {
25+
Smc *regs;
26+
uint32_t periph_id;
27+
28+
size_t banks_len;
29+
const struct memc_smc_bank_config *banks;
30+
31+
const struct pinctrl_dev_config *pcfg;
32+
};
33+
34+
static int memc_smc_init(const struct device *dev)
35+
{
36+
int ret;
37+
const struct memc_smc_config *cfg = dev->config;
38+
SmcCs_number *bank;
39+
40+
soc_pmc_peripheral_enable(cfg->periph_id);
41+
42+
ret = pinctrl_apply_state(cfg->pcfg, PINCTRL_STATE_DEFAULT);
43+
if (ret < 0) {
44+
return ret;
45+
}
46+
47+
for (size_t i = 0U; i < cfg->banks_len; i++) {
48+
if (cfg->banks[i].cs >= SMCCS_NUMBER_NUMBER) {
49+
return -EINVAL;
50+
}
51+
52+
bank = &cfg->regs->SMC_CS_NUMBER[cfg->banks[i].cs];
53+
54+
bank->SMC_SETUP = cfg->banks[i].setup_timing;
55+
bank->SMC_PULSE = cfg->banks[i].pulse_timing;
56+
bank->SMC_CYCLE = cfg->banks[i].cycle_timing;
57+
bank->SMC_MODE = cfg->banks[i].mode;
58+
}
59+
60+
return 0;
61+
}
62+
63+
#define SETUP_TIMING(node_id) \
64+
SMC_SETUP_NWE_SETUP(DT_PROP_BY_IDX(node_id, atmel_smc_setup_timing, 0)) \
65+
| SMC_SETUP_NCS_WR_SETUP(DT_PROP_BY_IDX(node_id, atmel_smc_setup_timing, 1)) \
66+
| SMC_SETUP_NRD_SETUP(DT_PROP_BY_IDX(node_id, atmel_smc_setup_timing, 2)) \
67+
| SMC_SETUP_NCS_RD_SETUP(DT_PROP_BY_IDX(node_id, atmel_smc_setup_timing, 3))
68+
#define PULSE_TIMING(node_id) \
69+
SMC_PULSE_NWE_PULSE(DT_PROP_BY_IDX(node_id, atmel_smc_pulse_timing, 0)) \
70+
| SMC_PULSE_NCS_WR_PULSE(DT_PROP_BY_IDX(node_id, atmel_smc_pulse_timing, 1)) \
71+
| SMC_PULSE_NRD_PULSE(DT_PROP_BY_IDX(node_id, atmel_smc_pulse_timing, 2)) \
72+
| SMC_PULSE_NCS_RD_PULSE(DT_PROP_BY_IDX(node_id, atmel_smc_pulse_timing, 3))
73+
#define CYCLE_TIMING(node_id) \
74+
SMC_CYCLE_NWE_CYCLE(DT_PROP_BY_IDX(node_id, atmel_smc_cycle_timing, 0)) \
75+
| SMC_CYCLE_NRD_CYCLE(DT_PROP_BY_IDX(node_id, atmel_smc_cycle_timing, 1))
76+
77+
#define BANK_CONFIG(node_id) \
78+
{ \
79+
.cs = DT_REG_ADDR(node_id), \
80+
.mode = COND_CODE_1(DT_ENUM_IDX(node_id, atmel_smc_write_mode), \
81+
(SMC_MODE_WRITE_MODE), (0)) \
82+
| COND_CODE_1(DT_ENUM_IDX(node_id, atmel_smc_read_mode), \
83+
(SMC_MODE_READ_MODE), (0)), \
84+
.setup_timing = SETUP_TIMING(node_id), \
85+
.pulse_timing = PULSE_TIMING(node_id), \
86+
.cycle_timing = CYCLE_TIMING(node_id), \
87+
},
88+
89+
#define MEMC_SMC_DEFINE(inst) \
90+
static const struct memc_smc_bank_config smc_bank_config_##inst[] = { \
91+
DT_INST_FOREACH_CHILD(inst, BANK_CONFIG) \
92+
}; \
93+
PINCTRL_DT_INST_DEFINE(inst); \
94+
static const struct memc_smc_config smc_config_##inst = { \
95+
.regs = (Smc *)DT_INST_REG_ADDR(inst), \
96+
.periph_id = DT_INST_PROP(inst, peripheral_id), \
97+
.banks_len = ARRAY_SIZE(smc_bank_config_##inst), \
98+
.banks = smc_bank_config_##inst, \
99+
.pcfg = PINCTRL_DT_INST_DEV_CONFIG_GET(inst), \
100+
}; \
101+
DEVICE_DT_INST_DEFINE(inst, memc_smc_init, NULL, NULL, \
102+
&smc_config_##inst, POST_KERNEL, \
103+
CONFIG_MEMC_INIT_PRIORITY, NULL);
104+
105+
DT_INST_FOREACH_STATUS_OKAY(MEMC_SMC_DEFINE)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,152 @@
1+
# Copyright (c) 2022, Basalte bv
2+
# SPDX-License-Identifier: Apache-2.0
3+
4+
description: |
5+
Atmel Static Memory Controller (SMC).
6+
7+
The SMC allows to interface with static-memory mapped external devices such as
8+
SRAM, PSRAM, PROM, EPROM, EEPROM, LCD Module, NOR Flash and NAND Flash.
9+
10+
The SMC is clocked through the Master Clock (MCK) which is controlled by the
11+
Power Management Controller (PMC).
12+
13+
The SMC controller can have up to 4 children defining the connected external
14+
memory devices. The reg property is set to the device's Chip Select.
15+
Device Tree example taken from the sam4_xplained board:
16+
17+
&smc {
18+
status = "okay";
19+
pinctrl-0 = <&smc_default>;
20+
pinctrl-names = "default";
21+
22+
is66wv51216dbll@0 {
23+
reg = <0>;
24+
25+
atmel,smc-write-mode = "nwe";
26+
atmel,smc-read-mode = "nrd";
27+
atmel,smc-setup-timing = <1 1 1 1>;
28+
atmel,smc-pulse-timing = <6 6 6 6>;
29+
atmel,smc-cycle-timing = <7 7>;
30+
};
31+
};
32+
33+
The above example configures a is66wv51216dbll-55 device. The device is a
34+
low power static RAM which uses NWE and NRD signals connected to the WE
35+
and OE inputs respectively. Assuming that MCK is 120MHz (cpu at full speed)
36+
each MCK cycle will be equivalent to 8ns. Since the memory full cycle is
37+
55ns, as per specification, it requires atmel,smc-cycle-timing of at least
38+
7 pulses (56ns). The atmel,smc-cycle-timing is composed of three parts:
39+
setup, pulse and hold. The setup is used to address the memory. The pulse
40+
is the time used to read/write. The hold is used to release memory. For the
41+
is66wv51216dbll-55 a minimum setup of 5ns (1 cycle) with at least 45ns
42+
(6 cycles) for CPU read/write and no hold time is required.
43+
Note: Since no hold parameter is available at SMC the atmel,smc-cycle-timing
44+
should have additional cycles to accommodate for hold values.
45+
46+
No Hold Time:
47+
cycle-timing (7) = setup (1) + pulse (6) + hold (0)
48+
49+
With 3 Hold Times:
50+
cycle-timing (10) = setup (1) + pulse (6) + hold (3)
51+
52+
Finally, in order to make the memory available you will need to define new
53+
memory device/s in DeviceTree:
54+
55+
sram1: sram@60000000 {
56+
compatible = "zephyr,memory-region", "mmio-sram";
57+
device_type = "memory";
58+
reg = <0x60000000 DT_SIZE_K(512)>;
59+
zephyr,memory-region = "SRAM1";
60+
};
61+
62+
compatible: "atmel,sam-smc"
63+
64+
include: [base.yaml, pinctrl-device.yaml]
65+
66+
properties:
67+
reg:
68+
required: true
69+
70+
"#address-cells":
71+
required: true
72+
const: 1
73+
74+
"#size-cells":
75+
required: true
76+
const: 0
77+
78+
peripheral-id:
79+
type: int
80+
description: peripheral ID
81+
required: true
82+
83+
child-binding:
84+
description: |
85+
Child device nodes are representing devices connected to the EBI/SMC bus.
86+
87+
properties:
88+
reg:
89+
type: int
90+
required: true
91+
description: |
92+
The device's SMC Chip Select number.
93+
Valid range: 0 - 3
94+
95+
atmel,smc-write-mode:
96+
type: string
97+
required: true
98+
description: |
99+
Select which signal is used for the write operation, either the chip
100+
select (ncs) or a dedicated write enable pin (nwe). The data is put
101+
on the bus during the pulse and hold steps of that signal.
102+
The internal data buffers are switched to output mode after the NCS_WR
103+
or NWE setup time.
104+
enum:
105+
- "ncs"
106+
- "nwe"
107+
108+
atmel,smc-read-mode:
109+
type: string
110+
required: true
111+
description: |
112+
Select which signal is used for the read operation, either the chip
113+
select (ncs) or a dedicated output enable pin (nrd). The data is read
114+
from the bus during the pulse and hold steps of that signal.
115+
enum:
116+
- "ncs"
117+
- "nrd"
118+
119+
atmel,smc-setup-timing:
120+
type: array
121+
required: true
122+
description: |
123+
This value is used to setup memory region (set address). The setup
124+
values is an array of the signals NWE, NCS_WR, NRD and NCS_RD
125+
where each value is configured in terms of MCK cycles. The SMC
126+
controller allows use of setups value of 0 (no delays) when
127+
consecutive reads/writes are used. Each value is encoded in
128+
6 bits where the highest bit adds an offset of 128 cycles.
129+
The effective value for each element is: 128 x setup[5] + setup[4:0]
130+
131+
atmel,smc-pulse-timing:
132+
type: array
133+
required: true
134+
description: |
135+
This value is used to effectivelly read/write at memory region (pulse phase).
136+
The pulse value is an array of the signals NWE, NCS_WR, NRD and NCS_RD where
137+
each value is configured in terms of MCK cycles and a value of 0 is forbidden.
138+
Each value is encoded in 7 bits where the highest bit adds an offset of 256
139+
cycles. The effective value for each element is: 256 x setup[6] + setup[5:0]
140+
141+
atmel,smc-cycle-timing:
142+
type: array
143+
required: true
144+
description: |
145+
SMC timing configurations in cycles for the total write and read
146+
length respectively.
147+
This value describes the entire write/read operation timing which
148+
is defined as: cycle = setup + pulse + hold
149+
Value has to be greater or equal to setup + pulse timing and
150+
is encoded in 9 bits where the two highest bits are multiplied
151+
with an offset of 256.
152+
Effective value for each element: 256 x cycle[8:7] + cycle[6:0]

0 commit comments

Comments
 (0)