

# Generated Specification Document

## INTRODUCTION

-- Overview --

sdrc\_core is a single-data-rate SDRAM controller that exposes a 32-bit application interface and drives a commodity SDR SDRAM with a configurable bus width (32/16/8). It accepts read/write requests, normalizes them to the SDR bus width, maps linear addresses to bank/row/column (via cfg\_col\_bits), and splits requests at page boundaries. Internally, per-bank scheduling and open-page control sequence PRE/ACT/READ/WRITE while enforcing device timing (tRP, tRCD, tRAS, tRC) and turnaround rules. An arbitration layer prioritizes initialization and periodic refresh over backend transfers, encodes active-low SDR commands (sdr\_cmd={CSn,RASn,CASn,WEn}), and manages acceptance windows (rdok, wrok, act\_ok, pre\_ok) and CKE power-down when idle. The read path captures DQ on the SDRAM clock, transfers it into the core clock aligned to CAS latency, and only signals application-side read valid when a full 32-bit word is assembled. The write path slices 32-bit words into bus-width beats (LSB-first), applies byte enables via DQM, asserts output enable (sdr\_den\_n active-low) during valid beats, and only advances application data when a complete 32-bit word is consumed. The controller performs power-up init (PRECHARGE ALL, AUTO-REFRESH, LOAD MODE REGISTER) and periodic refresh, and requires configuration of sdr\_width, cfg\_col\_bits, CAS latency, timing parameters, and mode register to match the SDRAM device. Integration notes: sdr\_cmd is active-low; observe rdok/wrok/act\_ok/pre\_ok before issuing backend commands; respect read/write turnaround rules; and constrain the read CDC path between pad\_clk and clk.

-- Key Features --

- 32-bit application to SDR SDRAM (8/16/32-bit) bridge with automatic bus-width conversion.
- Programmable address mapping (bank/row/column via cfg\_col\_bits); splits bursts at page boundaries; wrap-burst support.
- Width-normalized lengths/addresses for uniform downstream handling across sdr\_width modes.
- Per-bank open-page scheduler with admission FIFO; preserves front-end ordering; head-of-line dispatch.
- Per-bank FSM enforces tRP/tRCD/tRAS, tracks open rows, sequences PRE/ACT/RD/WR; page-hit optimization.
- Transfer/timing engine prioritizes init/refresh over read/write; drives CMD/ADDR/CKE; generates DQM and DQ output-enable.
- CAS-latency-aligned read pipeline (rdstart/rdok/rdlst) and write strobes (wrstart/wrnext/wrlst) with turnaround control.
- Precharge permission gating per bank and global ACT spacing guard (tRC); precharge only when tRAS satisfied.
- Integrated management: power-up (PRECHARGE ALL, multiple AUTO-REFRESH, LMR), periodic refresh (configurable), refresh activity signaling; idle power-down via CKE.
- Clean handshakes and flow control: front-end req/ack, back-end x2b\_ack; readiness windows x2b\_rdok/wrok/act\_ok; transaction ID/last/wrap propagation.
- Safe read-data capture across pad\_clk to clk via two-stage sync aligned to CAS latency; suited for related clocks.

- Bus-width conversion details: write slicing with byte-enable mapping; read aggregation emits full 32-bit words; pass-through in 32-bit mode.
- Configurability: sdr\_width, cfg\_col\_bits, CAS latency, tRP, tRC/tRCD, refresh period/max rows, mode register, enable.
- Reliability: FIFO under/overflow guards; one-bank-at-a-time issue; active-low sdr\_cmd {CS,RAS,CAS,WE} per SDRAM truth table.

-- Block Diagram and Submodule Summary --

- Block Diagram (textual)
- Application-side (32-bit) interface
- Requests: addr/len/id/rd-wr/wrap enter sdrc\_req\_gen; acceptance via req\_ack.
- Data: 32-bit write/read boundary through sdrc\_bs\_convert; byte enables active-low.
- Front-end request path
- sdrc\_req\_gen: normalizes widths to SDR beats, maps linear addr to bank/row/column, splits at page boundaries, emits r2b\_\* bursts.
- sdrc\_bank\_ctl: admits requests per bank, queues bank indices, arbitrates head-of-queue, multiplexes per-bank metadata/commands to back-end; implements four sdrc\_bank\_fsm instances.
- Command/transfer back-end
- sdrc\_xfr\_ctl: arbitrates management (init/refresh) vs bank commands, sequences PRE/ACT/READ/WRITE, enforces timing, aligns read data to CAS. Drives SDRAM pins: sdr\_cmd{CSn,RASn,CASn,WE}, sdr\_addr, sdr\_ba, sdr\_cke, sdr\_dqm, sdr\_den\_n, sdr\_dout; captures sdr\_din.
- Data width adapter and app handshakes
- sdrc\_bs\_convert: converts between 32-bit app data and 32/16/8-bit SDR bus beats; handles per-byte enables, serializes writes, assembles reads; coordinates via x2a\_\*/a2x\_\* strobes from sdrc\_xfr\_ctl.
- Read data capture
- pad\_sdr\_din captured on pad\_clk, re-registered on clk, then delivered to sdrc\_bs\_convert; app\_rd\_valid asserted when a full 32-bit word assembled.
- Major handshakes and gates
- Front-end: req -> req\_ack; r2b\_req -> b2r\_ack (per-bank/FIFO space).
- Bank/back-end: i2x\_req (per-bank ready) -> b2x\_req/b2x\_\* (selected bank) -> x2b\_ack.
- Back-end acceptance: x2b\_rdk, x2b\_wrok, x2b\_act\_ok, x2b\_pre\_ok[3:0]; global tRAS gate via b2x\_tras\_ok.
- Data strobes: x2a\_wrstart/wrnext/wrlast and x2a\_rdstart/rdok/rdlst bound app\_wr\_next/app\_last\_wr and app\_rd\_valid/app\_last\_rd.
- Configuration/control
- Data width via sdr\_width (32/16/8).
- Address mapping via cfg\_col\_bits (column width 8/9/10/11).
- Timing via cas\_latency and tRP/tRCD/tRAS/tRC/tMRD; refresh via rfsh\_time/rfsh\_rmax; enable via sdram\_enable.
- Submodule Summary
- sdrc\_req\_gen
  - Role: Front-end request splitter and address mapper; normalizes to SDR bus width, maps linear addresses to bank/row/col, prevents page crossing by splitting into up to two sub-requests (r2b\_last marks final).
  - Key I/O
    - Inputs: req, req\_addr, req\_len, req\_id, req\_wr\_n, req\_wrap, cfg\_col\_bits, sdr\_width, b2r\_arb\_ok.
    - Outputs: req\_ack; r2b\_req, r2b\_write, r2b\_req\_id, r2b\_ba, r2b\_raddr, r2b\_caddr, r2b\_len, r2b\_last, r2b\_wrap.
    - Handshake: r2b\_req -> b2r\_ack (per-bank admission).

- `sdrc_bank_ctl`
  - Role: Per-bank admission, scheduling, and muxing; maintains per-bank queues, arbitrates the active bank, forwards commands/meta to the back-end; exposes global idle and tRAS gating.
  - Key I/O
  - Inputs: `r2b_*` from `sdrc_req_gen`; `x2b_ack`, `x2b_rdok`, `x2b_wrok`, `x2b_act_ok`, `x2b_pre_ok[3:0]` from `sdrc_xfr_ctl`.
  - Outputs: `b2r_ack` to `sdrc_req_gen`; `b2x_req`, `b2x_cmd`, `b2x_ba`, `b2x_raddr`, `b2x_caddr`, `b2x_len`, `b2x_write`, `b2x_wrap`, `b2x_last` to `sdrc_xfr_ctl`; `b2x_idle`, `b2x_tras_ok`.
  - Children: `sdrc_bank_fsm[3:0]` (one per SDRAM bank).
  - `sdrc_bank_fsm` (instantiated per bank)
  - Role: Open-page bank machine; tracks open row, sequences PRE/ACT/READ/WRITE, enforces tRP/tRCD/tRAS, optimizes page hits, handles burst close on last beat.
  - Key I/O
  - Inputs: `r2b_*` for this bank; `x2b_ack`, `x2b_rdok`, `x2b_wrok`, `x2b_act_ok`, `x2b_pre_ok[bank]`; refresh awareness from `sdrc_xfr_ctl`.
  - Outputs: `i2x_req` (bank ready), `b2x_cmd/meta/addr/len/write/wrap/last` to `sdrc_xfr_ctl`.
  - `sdrc_xfr_ctl`
    - Role: Command/transfer engine and SDRAM pin driver; arbitrates management vs backend, sequences SDRAM commands, aligns read data to CAS latency, controls DQ enable and DQM; performs initialization and periodic refresh.
    - Key I/O
    - Inputs: `b2x_*` from `sdrc_bank_ctl`; configuration: `cas_latency`, timing registers (tRP/tRCD/tRAS/tRC/tMRD), `rfsh_time`, `rfsh_rmax`, `sdram_enable`; `pad_sdr_din/pad_clk`.
    - Outputs: `x2b_ack`, `x2b_rdok`, `x2b_wrok`, `x2b_act_ok`, `x2b_pre_ok[3:0]`; `x2a_wrstart/wrnext/wrlast`, `x2a_rdstart/rdok/rdlast`; `sdr_cmd`, `sdr_addr`, `sdr_ba`, `sdr_cke`, `sdr_dqm`, `sdr_den_n`, `sdr_dout`.
    - `sdrc_bs_convert`
      - Role: Width adapter between 32-bit app and 32/16/8-bit SDR bus; slices/serializes 32-bit writes to bus beats; converts byte-enables to `sdr_dqm`; assembles returning narrow beats into 32-bit words and generates `app_rd_valid`.
      - Key I/O
      - Inputs: `app_wr_data`, `app_wr_en_n`, `sdr_width`; `x2a_rdstart/rdok/rdlast` and `x2a_rddt` from `sdrc_xfr_ctl`.
      - Outputs: `a2x_wrdt`, `a2x_wren_n` to `sdrc_xfr_ctl`; `app_wr_next`, `app_last_wr`; `app_rd_data`, `app_rd_valid`, `app_last_rd`.
      - Data ordering: 16-bit mode low half then high half; 8-bit mode bytes [7:0]→[15:8]→[23:16]→[31:24]; `app_rd_data` presents full word on final beat.

-- Standards and Protocol Compliance --

Implements the JEDEC single data rate (SDR) SDRAM standard for command set, signaling, addressing, timing, initialization, refresh, and power management. Commands are driven using active-low CSn/RASn/CASn/WEn per the JEDEC truth table, with BA/A multiplexed addressing (two BA bits) and A10 semantics for PRECHARGE ALL. Supported commands: ACTIVATE, READ, WRITE, PRECHARGE (bank/all), AUTOREFRESH, LOAD MODE REGISTER, and BURST TERMINATE; auto-precharge is not used (explicit PRECHARGE only). Timing conformance is enforced via configurable parameters for tRP, tRCD, tRAS, tMRD, and tRC, plus activation spacing guards and CAS-latency-aligned read pipelines; read→write turnarounds are controlled, and integrators must set device-specific constraints (e.g., tWR, tRRD, tRFC, tCCD, tWTR/tRTW) to match the target SDRAM. Initialization follows JEDEC: PRECHARGE ALL, a programmable number of AUTOREFRESH cycles spaced by tRC, then LOAD MODE REGISTER with tMRD recovery; normal operation begins after an init-done indication. Refresh compliance is achieved via a programmable timer/counter (`rfsh_time`/`rfsh_rmax`) that schedules AUTOREFRESH with required precharge and tRC spacing to meet row retention. Data bus compliance: x8/x16/x32 widths with DQM-based byte/halfword masking

per JEDEC; DQ drivers are enabled only during valid write beats, and read data is captured per the configured CAS latency. Burst behavior adheres to the mode register's burst length/type and CAS latency; BURST TERMINATE is issued when ending transfers early. CKE is used per JEDEC for power-down and active operation. Interoperability limits: designed for SDR SDRAM with two BA lines; not intended for DDR/DDR2+. Correct operation requires accurate mapping of active-low signals, A10 behavior, mode register image, and timing/CDC constraints at the SDRAM interface.

#### -- SDRAM Command Encoding (Active-Low CS/RAS/CAS/WE) --

sdr\_cmd[3:0] is the SDRAM command bus with bit order {CSn, RASn, CASn, WEn}; all signals are active-low (logic 0 asserts). The discrete pins sdr\_cs\_n/sdr\_ras\_n/sdr\_cas\_n/sdr\_we\_n mirror this bus. Encodings used by the design:

- PRECHARGE: 0010; single-bank when sdr\_addr[A10]=0.
- PRECHARGE ALL: 0010 with sdr\_addr[A10]=1 (bank value ignored).
- ACTIVATE: 0011.
- READ: 0101.
- WRITE: 0100.
- AUTO-REFRESH: 0001.
- LOAD MODE REGISTER: 0000; mode/address presented on sdr\_ba/sdr\_addr.
- BURST TERMINATE: 0110.

Address/bank semantics: ACTIVATE uses sdr\_ba as bank and sdr\_addr as row; READ/WRITE use sdr\_ba as bank and sdr\_addr as column. Commands are encoded and arbitrated solely by sdrc\_xfr\_ctl: it maps b2x\_cmd (00 PRE, 01 ACT, 10 READ, 11 WRITE) to sdr\_cmd and injects management/initialization/refresh commands. The SDRAM samples sdr\_addr/sdr\_ba only when CSn is asserted (CSn=0). Ensure board pin mapping respects the {CSn, RASn, CASn, WEn} order.

#### -- Clocking and Reset Domains --

- Clocks: Two domains are used. clk is the core/application clock; all controller logic (request generation, bank control/FSM, transfer control, bus-width conversion, command arbitration, CAS-latency pipeline, init/refresh timers, and sdr\_cke control) runs in clk. pad\_clk is the SDRAM pad-side clock used to capture incoming DQ during reads.
- Reset: reset\_n is a global, active-low reset distributed to all submodules. Treat reset\_n as synchronous to clk unless otherwise indicated by RTL. Upon reset, all FSMs, counters, FIFOs, pipelines, and accumulators are cleared. sdrc\_xfr\_ctl additionally clears initialization/refresh machinery and holds sdr\_init\_done low until the sdram\_enable-driven initialization sequence completes.
- Domain assignment: sdrc\_req\_gen, sdrc\_bank\_ctl, sdrc\_bank\_fsm, sdrc\_xfr\_ctl, and sdrc\_bs\_convert are clocked by clk and use reset\_n. Per-bank timing gates (tRP/tRCD/tRAS), row tracking, ACT spacing (x2b\_act\_ok), turnaround gating (rd\_pipe\_mt), and precharge permission (b2x\_tras\_ok, cb\_pre\_ok) are generated and consumed within clk.
- Read CDC: The read data return path crosses from pad\_clk to clk via a two-stage capture: pad\_sdr\_din1 is registered on pad\_clk; pad\_sdr\_din2 is registered on clk. There is no handshake on this path; clk and pad\_clk must be related (identical or phase-aligned/derived). Both capture stages are reset by reset\_n. CAS-latency alignment and app-side validity strobes are generated after the crossing in clk.
- SDRAM outputs: SDRAM command/address/data outputs are generated by clk logic. If an external PHY or I/O wrapper re-registers these signals on pad\_clk, ensure appropriate output timing; otherwise, constrain them directly from clk to the SDRAM.
- Constraints and usage: Declare pad\_clk as related/derived from clk and constrain the pad\_clk→clk read CDC; apply appropriate exceptions (e.g., multicycle or false-path) to match implementation. Do not use the two-flop CDC with fully asynchronous, unrelated clocks without adding synchronization. If pad\_clk == clk, the design operates as single-clock with registered input capture. If pad\_clk is a phase-related derivative, maintain the two-flop CDC and timing constraints.

- Reset effects (examples): `sdrc_bs_convert` clears `wr_xfr_count/rd_xfr_count` and `saved_rd_data`; `sdrc_bank_fsm` clears `bank_valid` and tRP/tRCD/tRAS timers and row tracking; `sdrc_bank_ctl` resets rank FIFO depth/contents and selection; `sdrc_xfr_ctl` clears the read pipeline, timers (`tmr0, cntr1, rfsh_timer`), and returns `sdr_cke` to the idle policy. CAS-latency pipeline flops are reset to a known state.
- Application interface: All app-side handshakes (`req/ack, app_wr_next, app_rd_valid`) are in `clk`; no CDC is required on the application interface.

-- Clock Domain Crossing Strategy and Assumptions --

#### Clock Domain Crossing Strategy and Assumptions

- Clock domains
- `clk`: Core/application domain for all controller FSMs and datapath (`sdrc_req_gen, sdrc_bank_ctl/sdrc_bank_fsm, sdrc_xfr_ctl, sdrc_bs_convert`).
- `pad_clk`: SDRAM pad/device domain used to capture incoming DQ.
- Read data CDC (`pad_clk → clk`)
  - Two-stage related-clock transfer:
    - `pad_sdr_din1 <= pad_sdr_din` sampled on `pad_clk` (pad domain capture).
    - `pad_sdr_din2 <= pad_sdr_din1` retimed on `clk` (core domain).
  - Post-crossing read-valid/start/last strobes (e.g., `x2a_rdok/x2a_rdstart/x2a_rdlas`t) are generated in `clk`, aligned via the CAS-latency pipeline after the retime.
  - Assumption: `clk` and `pad_clk` are frequency/phase related (same source or known relationship). The multi-bit DQ bus relies on timing closure, not asynchronous synchronizers.
- Write/command path (`clk → pad_clk`)
  - `sdr_cmd[CSn,RASn,CASn,WEn], sdr_addr, sdr_ba, sdr_cke, sdr_dout, sdr_dqm, sdr_den_n` are driven in `clk` and assumed to be delivered synchronously to the SDRAM/pad domain (`pad_clk`) via IO/PHY timing.
  - No explicit CDC synchronizers are used; correctness depends on the `clk ↔ pad_clk` relationship and meeting SDRAM setup/hold at the device.
- No other CDCs
- All internal handshakes (`r2b_*, b2x_*, x2b_*, x2a_*`) remain in `clk`; no cross-domain synchronization required for these.
- Constraints and timing assumptions
  - Constrain `pad_clk → clk` retiming path for DQ as related clocks (define clock relationships or apply appropriate multicycle/latency constraints consistent with CAS/data capture).
  - Constrain `clk → pad_clk` launch for command/address/write data to meet SDRAM setup/hold at the IO boundary (account for clock forwarding/phase).
  - Do not treat the multi-bit read bus as asynchronous; timing closure is required across the two-stage capture.
- Integration guidance for unrelated clocks
  - If `clk` and `pad_clk` are not frequency/phase related, this core's scheme is not sufficient: add a pad-side PHY for write/command and an asynchronous read data CDC (e.g., FIFO) external to the core. The current design does not include such asynchronous CDC structures.
- Reset considerations
  - `reset_n` usage across domains is not defined here; ensure domain-appropriate, synchronously deasserted resets in each clock domain at the system level if a common reset spans `clk` and `pad_clk`.

- Read pipeline alignment
- CAS-latency alignment logic in clk assumes a deterministic  $\text{clk} \leftrightarrow \text{pad\_clk}$  relationship so  $\text{x2a\_rdok}$  aligns with valid  $\text{pad\_sdr\_din2}$  data after the retiming stage.

-- Design Assumptions and Operating Conditions --

- Target device: JEDEC-compliant single data rate (SDR) SDRAM with 4 banks (BA[1:0]). Commands use active-low pins {CS\_n, RAS\_n, CAS\_n, WE\_n}. PRECHARGE ALL requires A10=1; per-bank precharge uses A10=0.
- Clocks: Two related domains are required:  $\text{pad\_clk}$  (SDRAM I/O/PHY) and  $\text{clk}$  (core/app). They must be frequency-related (mesochronous/synchronous) and constrained for CDC. Read data crosses  $\text{pad\_clk} \rightarrow \text{clk}$  via two-stage capture; ensure STA exceptions or phase alignment as needed.
- Reset and enables:  $\text{reset\_n}$  is active-low and must asynchronously/synchronously place the controller and SDRAM in a known state.  $\text{sdram\_enable}$  must be asserted to run/maintain initialization; configuration changes require re-initialization.
- Data widths: Application interface is fixed 32-bit. SDRAM DQ bus width is configurable to 32/16/8 bits ( $\text{sdr\_width}$ ). Bus-width conversion serializes/deserializes across narrow beats; app must honor  $\text{app\_wr\_next}$  (advance only when high) and  $\text{app\_rd\_valid}$  (consume data only when high).
- Address mapping: Linear app addresses map to {ROW, BA, COL};  $\text{cfg\_col\_bits}$  selects column width of 8/9/10/11 bits (page size 256/512/1024/2048 x8-beats). BA are the two bits above the column field; ROW is above BA. Requests are normalized to x8-beat units based on  $\text{sdr\_width}$ .
- Request splitting: Transfers that would cross a page boundary are split into at most two sub-requests unless  $\text{wrap}=1$ . Splits respect remaining space in the current page.
- SDRAM timings: tRP, tRCD, tRAS, tRC (trcar\_delay), and tMRD must be programmed per device datasheet at the intended frequency/voltage/temperature.  $\text{x2b\_act\_ok}$  enforces ACTIVATE spacing; precharge\_ok requires  $\text{cb\_pre\_ok} \& \text{b2x\_tras\_ok}$  to honor tRAS.
- CAS latency:  $\text{cas\_latency}$  must match the SDRAM mode register; it sets read-return pipeline tap points. Misconfiguration will misalign read timing ( $\text{rd\_ok}/\text{rdstart}/\text{rdlast}$ ) and is not tolerated.
- Initialization: Requires PRECHARGE ALL, a programmed number of AUTO-REFRESH cycles spaced by tRC, then LOAD MODE REGISTER with  $\text{sdram\_mode\_reg}$ .  $\text{sdr\_init\_done}$  must assert before backend acceptance.
- Refresh: A free-running accumulator issues AUTO-REFRESH when  $\text{rfsh\_row\_cnt}$  reaches  $\text{rfsh\_rmax}$ , with tRC spacing.  $\text{rfsh\_time}$  and  $\text{rfsh\_rmax}$  must meet the SDRAM tREF requirement over all rows and operating conditions.
- Arbitration/backpressure: Initialization/refresh have priority over backend reads/writes. Issue READ/WRITE only when  $\text{x2b\_rdok}/\text{x2b\_wrok}$  are asserted and  $\text{x2b\_act\_ok}$  permits. Admission respects per-bank queues (rank FIFO depth up to 4);  $\text{b2r\_arb\_ok}$  indicates capacity.
- Turnaround policy: Writes are blocked while the read-return pipeline is non-empty ( $\text{rd\_pipe\_mt}=0$ ) to avoid bus turnaround hazards. Precharge permissions ( $\text{x2b\_pre\_ok}$ ) are suppressed during active streaming.
- Burst behavior: Non-wrap transfers may be terminated using BURST TERMINATE at sub-burst boundaries; last.WRAP metadata must be consistent with desired termination and page limits.
- Power-down:  $\text{sdr\_cke}$  may be deasserted only when the controller is fully idle (no init/refresh/backend/read-return activity). The SDRAM must support CKE power-down; wake-up timing must be satisfied by configuration.
- Board-level mapping: External wiring must match SDRAM truth table for command bit order and active-low polarity. DQM width/lane mapping must correspond to  $\text{sdr\_width}$  and PCB byte-lane wiring. Byte-enables/masks are active-low and mirror  $\text{app\_wr\_en\_n}$  during writes.
- Configuration stability: Operating parameters ( $\text{sdr\_width}$ ,  $\text{cfg\_col\_bits}$ ,  $\text{cas\_latency}$ , timing delays,  $\text{sdram\_mode\_reg}$ ) are expected stable during normal operation. Any change requires a full re-initialization sequence.

- Electrical/timing conditions: pad\_clk-driven signals must meet SDRAM setup/hold at the target frequency. Constrain pad\_clk $\leftrightarrow$ clk CDC paths (read capture) appropriately. Ensure supply, temperature, and clock quality stay within the SDRAM datasheet operating ranges used to derive timing parameters.

## IO PORTS

-- Port Summary --

- Clocks and Reset

- clk (input): Core/application clock.
- pad\_clk (input): SDRAM pad/DQ capture clock.
- reset\_n (input, active-low): Asynchronous reset.

- Application Request Interface

- app\_req (input): New request strobe.
- app\_req\_addr[...] (input): Linear address (mapped to bank/row/column internally).
- app\_req\_len[...] (input): Burst length in 32-bit words (application side).
- app\_req\_wr\_n (input): 0=write, 1=read.
- app\_req\_id[...] (input): Transaction ID tag.
- app\_req\_wrap (input): Allow wrap within page/burst.
- app\_req\_ack (output): Acceptance strobe for app\_req.
- app\_req\_idle (output): High when controller can accept a new request.

- Application Write Data Interface

- app\_wr\_data[31:0] (input): Write data word.
- app\_wr\_en\_n[3:0] (input, active-low per byte): Byte enables for app\_wr\_data.
- app\_wr\_next (output): Advance to next 32-bit word after all narrow beats for the current word are issued on SDR bus.
- app\_last\_wr (output): Last-beat indicator forwarded from SDR side.

- Application Read Data Interface

- app\_rd\_data[31:0] (output): Assembled 32-bit read data.
- app\_rd\_valid (output): Asserted when app\_rd\_data contains a full valid word.
- app\_last\_rd (output): Last-beat indicator forwarded from SDR side.
- app\_rd\_id[...] (output): Transaction ID associated with valid read data.

- SDRAM Control/Address Pins

- sdr\_cmd[3:0] (output, active-low): Command bus {CS\_n, RAS\_n, CAS\_n, WE\_n}.
- sdr\_addr[...] (output): SDRAM address bus; drives row on ACT, column on READ/WRITE; PRECHARGE uses row with A10 per target (bank/all).
- sdr\_ba[1:0] (output): Bank address.
- sdr\_cke (output): SDRAM clock enable.
- sdr\_dqm[...] (output): Data mask/byte-lane enables (width depends on SDR data width).

- SDRAM Data Pins

- sdr\_dout[...] (output): Write data to SDRAM (8/16/32 bits selected by sdr\_width).

- `sdr_den_n` (output, active-low): DQ output enable (assert only during valid write beats).
- `pad_sdr_din[...]` (input): Read data from SDRAM pads (captured on `pad_clk`; width per `sdr_width`).

#### - Configuration and Timing

- `sdr_width[1:0]` (input): SDR data width select; 00=32-bit, 01=16-bit, others=8-bit.
- `cfg_col_bits[1:0]` (input): Column address width selector (8/9/10/11 bits).
- `cas_latency[...]` (input): CAS latency used to align read-valid strobes.
- `sdram_mode_reg[...]` (input): Mode register value for initialization.
- `sdram_enable` (input): Enables SDRAM, triggers power-up initialization.
- `trp_delay[...]` (input): PRECHARGE timing (tRP).
- `trcar_delay[...]` (input): tRC spacing parameter.
- `rfsh_time[...]` (input): Refresh interval.
- `rfsh_rmax[...]` (input): Maximum accumulated refresh count.

#### - Notes

- Widths marked [...] depend on device geometry or application protocol (e.g., `sdr_addr`, `cas_latency`, timing fields, `app_req_id`). The app side data width is fixed at 32 bits with 4 byte enables; SDR-side data and DQM widths depend on `sdr_width`. Read data crosses `pad_clk`→`clk` domains; constrain CDC accordingly.

#### -- Clock and Reset Signals --

Clocks: (1) `clk` — primary logic/application clock. All internal control, arbitration, bank/transfer FSMs, bus-width conversion, and inter-module handshakes (`r2b_*`, `b2x_*`, `x2b_*`, `x2a_*`) are synchronous to `clk`. (2) `pad_clk` — SDRAM pad/PHY clock. Used to time SDRAM command/address/data/mask outputs (`sdr_cmd`, `sdr_addr`, `sdr_ba`, `sdr_dqm`, `sdr_den_n`, `sdr_dout`) and to capture incoming read data at the pads.

Clock relationship: `clk` and `pad_clk` must be phase/frequency related. If `clk == pad_clk`, signals are naturally aligned. If different but related, define explicit launch/capture relationships in timing constraints (SDC/XDC). Read data crosses `pad_clk`→`clk` via a two-stage path: `sdr_din` → `pad_sdr_din1` (`pad_clk`) → `pad_sdr_din2` (`clk`). Treat this as a timed crossing between related clocks (no asynchronous synchronizer); constrain setup/hold or use multicycle as appropriate.

Reset: `reset_n` is active-low, asynchronously assertable, and must be synchronously deasserted in both `clk` and `pad_clk` domains to prevent partial-reset hazards. It resets all FSMs, counters, FIFOs, pipelines, and capture flops across domains, including: `sdrc_req_gen` (request-splitting FSM, address/length trackers), `sdrc_bank_ctl/bank_fsm` (per-bank state, timing trackers: tRP/tRCD/tRAS), `sdrc_xfr_ctl` (transfer/init FSMs, refresh counters, read-return pipeline; `sdr_cke` deasserted until init), `sdrc_bs_convert` (wr/rd transfer counters, saved read data), and pad-side registers (`pad_sdr_din1` in `pad_clk`, `pad_sdr_din2` in `clk`).

Constraints: Provide IO timing constraints on SDRAM pins relative to `pad_clk` and define the `clk`↔`pad_clk` relationship. Constrain the `pad_clk`→`clk` read-data capture path; apply CDC methodology consistent with related clocks.

Operational note: Hold application requests/handshakes until `sdr_init_done` is asserted after reset (initialization is triggered by `sdram_enable`).

#### -- Application Request Interface --

##### Overview

- The Application Request Interface is a 32-bit, synchronous front-end through which the application issues read and write bursts to SDR SDRAM. It exposes a control/request channel (address, length, attributes, handshake) and a 32-bit data channel with per-byte enables for writes and valid strobes for reads. All interface signals are synchronous to `clk`.

## Control/request channel

- Signals:
  - app\_req: Assert to present a request; hold high with fields stable until acknowledged.
  - app\_req\_ack: One-cycle acknowledge when the controller accepts the request.
  - app\_req\_addr: Linear word address (unit = 32-bit word). No additional alignment required by the app.
  - app\_req\_len: Burst length in 32-bit words.
  - app\_req\_wr\_n: Direction (0 = write, 1 = read).
  - app\_req\_id: Application-supplied transaction ID (width user-defined); latched on acceptance and propagated to the data phase.
  - app\_req\_wrap: Hint that the burst is wrapped; suppresses forced page split at a row/page boundary.
  - app\_req\_idle: Optional status indicating the controller is idle/ready to accept a new request.
- Handshake/use:
  - The controller asserts app\_req\_ack only when internal queues/timing allow; the app must hold app\_req and all request fields stable until app\_req\_ack.
  - Exactly one app\_req\_ack is returned per original request, even if the controller internally splits the burst at a page boundary.
- Addressing and burst/page behavior:
  - Addresses and lengths are specified in 32-bit word units and are normalized internally to the SDR bus width (x32/x16/x8).
  - At page (row/column) boundaries, the controller may split a request into at most two SDR-aligned sub-requests; admission order is preserved. app\_req\_wrap=1 hints that the app intends a wrapped burst and suppresses forced page splitting.
- Ordering and management:
  - Accepted requests are preserved in FIFO order across banks; service may interleave by bank readiness and timing.
  - During initialization/refresh or turnaround constraints, acceptance and command issuance may be throttled; backpressure is reflected via app\_req\_ack deassertion.

## Write data channel (application side, 32-bit)

- Signals:
  - app\_wr\_data[31:0]: Write data word.
  - app\_wr\_en\_n[3:0]: Active-low per-byte enables for app\_wr\_data.
  - app\_wr\_next: Advance strobe; pulses when the controller has consumed the SDR beats corresponding to one 32-bit word. The app must only present the next word when app\_wr\_next is asserted.
  - app\_last\_wr: Assert by the app to mark the last word of the current write burst; used to frame burst completion.
- Width conversion and ordering:
  - sdr\_width determines beats-per-word: x32=1 beat, x16=2 beats, x8=4 beats.
  - Byte enables are narrowed per SDR beat; beat ordering is increasing significance (LSB to MSB).
- Usage:
  - Hold app\_wr\_data/app\_wr\_en\_n stable and advance only on app\_wr\_next.
  - app\_last\_wr brackets burst completion and resets internal beat aggregation counters.

## Read data channel (application side, 32-bit)

- Signals:
  - app\_rd\_data[31:0]: Read data word.
  - app\_rd\_valid: Pulses when a full 32-bit word has been assembled from SDR beats; sample app\_rd\_data only when asserted.
  - app\_last\_rd: Asserted alongside the final word of the current read burst to bracket completion.
  - Returned ID: The transaction ID associated with the request is forwarded in-phase with read

start/valid strobes to support matching (width matches app\_req\_id).

- Width conversion and timing:
- sdr\_width governs aggregation: x32=1 beat, x16=2 beats, x8=4 beats.
- Read-valid timing is aligned to the configured CAS latency and the internal capture pipeline; latency is variable, but app\_rd\_valid provides precise data timing.

#### Configuration and timing

- sdr\_width: Selects SDR data bus width (32/16/8) and governs beat aggregation/serialization and app\_wr\_next/app\_rd\_valid generation.
- cfg\_col\_bits: Determines page/column width and influences where bursts may split across page boundaries.
- cas\_latency: Configures read-return alignment to the SDRAM; the app must tolerate variable latency and rely on app\_rd\_valid.
- Clocks and capture:
  - All app-side signals are synchronous to clk.
- Read data are captured from the SDRAM pad clock domain via a two-stage transfer; treat as a CDC unless clocks are phase-related.

#### Usage rules

- Issue one request at a time; wait for app\_req\_ack before issuing the next. Hold request fields stable until ack.
- For writes, present data/byte enables and advance only on app\_wr\_next; use app\_last\_wr to mark burst end.
- For reads, consume data only when app\_rd\_valid; app\_last\_rd marks burst end.
- IDs should be unique across outstanding requests if the application relies on ID-based matching.

#### Behavioral guarantees

- Admission order is maintained across banks; internal arbitration may interleave service by bank readiness.
- The app sees a single app\_req\_ack per original request, irrespective of internal page splits.
- Backpressure can occur due to queue limits, initialization/refresh, or turnaround hazards; handshake/data strobes reflect readiness.

-- Application Write Data Interface --

#### Application Write Data Interface

##### Purpose

- Provides a 32-bit, word-oriented data path from the application to the SDRAM controller. Internally serialized to match the configured SDRAM data bus width (32/16/8 bits) while preserving per-byte write enables.

##### Signals

- app\_wr\_data[31:0] (input): 32-bit write data word from the application.
- app\_wr\_en\_n[3:0] (input): Active-low per-byte write enables for app\_wr\_data bytes [7:0], [15:8], [23:16], [31:24].
- app\_wr\_next (output, pulse): Asserted for one cycle when the controller has consumed exactly one full 32-bit application word; advance the application write buffer only on this pulse.
- app\_last\_wr (output, pulse): Asserted on the last SDRAM data beat of the active write burst (end-of-burst indicator).
- sdr\_width (configuration): Selects SDR data bus width: 00=32-bit, 01=16-bit, others=8-bit; determines sub-beat serialization described below.

### Handshake and timing

- The application must hold app\_wr\_data and app\_wr\_en\_n stable until app\_wr\_next pulses. There is no separate ready/valid on the app side.
- app\_wr\_next pulses once per 32-bit application word, coincident with the final sub-beat that completes that word on the SDR bus.
- Inter-beat spacing and sub-beat pacing are controlled by the controller and may include stalls; the application observes only app\_wr\_next for word-level flow control.

### Width adaptation and ordering

- sdr\_width=32-bit: Pass-through. One SDR beat per app word; app\_wr\_next pulses every beat.
- sdr\_width=16-bit: Two sub-beats per app word.
- Sub-beat 0: app\_wr\_data[15:0] with enables app\_wr\_en\_n[1:0].
- Sub-beat 1: app\_wr\_data[31:16] with enables app\_wr\_en\_n[3:2].
- app\_wr\_next pulses on sub-beat 1.
- sdr\_width=8-bit: Four sub-beats per app word.
- Sub-beats 0/1/2/3 carry bytes [7:0]/[15:8]/[23:16]/[31:24] respectively; app\_wr\_en\_n bit for the active byte is applied.
- app\_wr\_next pulses on sub-beat 3.
- Narrow transfers emit lower-significance bytes/half-words first (LSB-first) toward the SDR bus.

### Byte enables

- app\_wr\_en\_n is active-low; a high value masks (disables) the corresponding byte on the SDR bus. Ensure polarity matches system convention.

### Burst and boundary behavior

- app\_last\_wr pulses on the final SDRAM data beat of the current write burst (may be mid-word for narrow buses if a burst ends early due to scheduling or page boundary splits).
- Internal counters realign to sub-beat 0 at reset and on app\_last\_wr, ensuring the next app word starts at the first sub-beat.

### Integration guidance

- Keep app\_wr\_data/app\_wr\_en\_n stable until app\_wr\_next; update to the next word only when app\_wr\_next=1.
- For simpler buffering, size bursts to whole 32-bit words; partial-word endings are handled but may require careful application-side bookkeeping.
- Expect occasional stalls between sub-beats due to SDRAM timing or read/write turnarounds; no additional app-side ready is provided beyond app\_wr\_next.

-- Application Read Data Interface --

Provides a 32-bit application-side read data stream independent of the physical SDR SDRAM bus width (32/16/8). Interface signals: app\_rd\_data[31:0] carries the 32-bit read word; app\_rd\_valid pulses high only when app\_rd\_data holds a complete 32-bit word (sample on this pulse; no ready/backpressure is provided); app\_last\_rd pulses on the last 32-bit word of the active read burst (page-split bursts produce multiple app\_last\_rd pulses). A transaction ID sideband (x2a\_id) is forwarded and remains constant for the duration of the burst to correlate responses with requests.

Width conversion: In 32-bit SDR mode, data and valid pass through one-for-one. In 16-bit SDR mode, two 16-bit beats are assembled per 32-bit word; app\_rd\_valid asserts on every second narrow beat; byte order is [15:0] then [31:16] (final beat forms the MS half-word). In 8-bit SDR mode, four 8-bit beats are assembled per 32-bit word; app\_rd\_valid asserts on every fourth narrow beat; byte order is [7:0] → [15:8] → [23:16] → [31:24] (final beat forms the MS byte). Data remains in-order across all modes.

Timing and latency: app\_rd\_valid is aligned to the SDRAM CAS latency via an internal pipeline in the transfer controller; configuration must match the SDRAM mode register to avoid misalignment. The narrow-beat valid drives the width assembly. Read/write turnaround is managed internally to preserve read timing and ordering.

Clocking and CDC: SDR pad data is captured on the pad clock, synchronized into the core/application clock, then width-converted before presenting app\_rd\_data/app\_rd\_valid. Treat this path as a CDC in timing constraints unless clocks are phase-related.

Reset: Internal assembly counters and pipelines reset on global reset and at burst boundaries; app\_rd\_valid is deasserted during reset. For 8/16-bit SDR widths, issue read lengths aligned to 32-bit words to avoid partial-word tails (assembly state resets at burst end).

#### -- SDRAM Device Interface (Command/Address/Data) --

The SDR SDRAM device is driven by an active-low command bus sdr\_cmd[3:0]={CSn,RASn,CASn,WE<sub>n</sub>}, an address bus sdr\_addr, bank selects sdr\_ba[1:0], and a bidirectional DQ bus with DQM masking. Commands (active-low encodings) are: PRECHARGE=0010, ACTIVATE=0011, READ=0101, WRITE=0100, AUTO-REFRESH=0001, LOAD MODE REGISTER=0000, BURST TERMINATE=0110. sdrc\_xfr\_ctl issues commands with priority mgmt/init/refresh > backend transfers > internal burst continuation. Addressing: ACTIVATE drives the row on sdr\_addr for the selected sdr\_ba; READ/WRITE drive the column. For PRECHARGE, A10 on sdr\_addr selects scope (0=bank-only, 1=all-banks). sdr\_addr and sdr\_ba are valid and latched while CSn=0 in the issuing cycle. Upstream sdrc\_req\_gen maps application addresses to {bank,row,column} based on cfg\_col\_bits and prevents page crossing. Data/DQM: The device DQ width is configurable (8/16/32 bits). sdrc\_bs\_convert adapts 32-bit application data to device beat order and byte-lane mapping. On writes, sdr\_dout drives data; sdr\_den\_n (active-low) is asserted only on valid write beats to enable DQ outputs; sdr\_dqm carries per-byte write masks (mirroring app-side enables) and is 0 during reads. On reads, DQ is tri-stated by the core; data is captured on the SDRAM clock and transferred to the core clock via a CAS-latency-aligned two-stage path, with rdstart/rdok/rdlast strobes generated by sdrc\_xfr\_ctl. Bursting: sdrc\_xfr\_ctl streams successive READ/WRITE column commands within a page and asserts BURST TERMINATE at the end of non-wrapping bursts.

#### -- Management and Configuration Inputs --

Management and configuration inputs define SDRAM geometry, timing, initialization, refresh policy, and bus-width adaptation. Inputs and behavior: sdram\_enable — Master enable for the controller's initialization/management FSM; assert to run the power-up sequence and keep asserted for normal operation; sdr\_init\_done is valid only while high. sdram\_mode\_reg — Mode Register value loaded during initialization (burst length, CAS latency, write-burst mode, etc.); must be coherent with cas\_latency and the target SDRAM. cas\_latency — CAS latency used to align the read-return pipeline; must match the CAS configured in sdram\_mode\_reg for correct data-valid strobes. sdr\_width[1:0] — SDRAM data bus width and width-conversion selection: 00=32-bit, 01=16-bit, else=8-bit; drives byte/beat conversion and address/length normalization. cfg\_col\_bits[1:0] — SDRAM column address width: 00=8, 01=9, 10=10, 11=11; determines page size (256/512/1024/2048 beats) and geometry mapping. trp\_delay — tRP in controller clock cycles; minimum PRECHARGE-to-ACTIVATE spacing enforced by bank and management FSMs. trcd\_delay — tRCD in cycles; ACTIVATE-to-READ/WRITE spacing enforced by bank FSM. tras\_delay — tRAS in cycles; minimum active-row time; precharge permission is gated by both streaming safety and tRAS satisfaction. trcar\_delay — Row-cycle/refresh spacing in cycles; used to separate AUTO-REFRESH commands in management sequences. tmrd\_delay — tMRD in cycles; wait time after LOAD MODE REGISTER before normal operation.

rfsh\_time — Refresh accumulation period in cycles; rfsh\_timer timeout interval used to accumulate refresh demand. rfsh\_rmax — Refresh accumulator threshold; number of AUTO-REFRESH commands to issue when a refresh sequence starts. a2b\_req\_depth — Per-bank rank FIFO depth (1..4) for scheduling; sets front-end backpressure window.

Usage and constraints: All timing delays are specified in controller clock cycles and must satisfy the SDRAM datasheet minima at the operating frequency. sdr\_width and cfg\_col\_bits are intended as static configuration; do not change while traffic is active. cas\_latency must be consistent with sdram\_mode\_reg to avoid misaligned read-valid strobes. Refresh policy is defined by rfsh\_time and rfsh\_rmax: each rfsh\_time timeout increments the refresh accumulator; when it reaches rfsh\_rmax, that many AUTO-REFRESH commands are issued. Precharge commands are allowed only when both streaming safety and tRAS requirements are met. Provide clock/CDC constraints for pad\_clk $\leftrightarrow$ clk paths; the controller assumes related clocks for command/data timing. Initialization order: PRECHARGE ALL (A10=1) → NxAUTO-REFRESH (spaced by trcar\_delay) → LOAD MODE REGISTER (wait tmrd\_delay) → sdr\_init\_done.

-- Status and Handshake Signals --

Status and handshake signaling across sdrc\_core and submodules:

- Application request (front-end)
- req: app raises a new request; req\_ack: one-cycle accept pulse from sdrc\_req\_gen when b2r\_arb\_ok is high.
- req\_idle: sdrc\_req\_gen idle (no pending segments); r2x\_idle: ready to accept a new request (idle and not mid-split).
- r2b\_req: sub-request strobe per page-aligned segment; r2b\_last: marks final sub-request of the original app request.
- r2b\_write, r2b\_req\_id, r2b\_len, r2b\_wrap: metadata latched and forwarded with r2b\_req.
- Per-bank admission and selection (sdrc\_bank\_ctl)
- b2r\_arb\_ok: backpressure to req\_gen (FIFO space available for new requests).
- b2r\_ack: accept pulse back to req\_gen when addressed bank captures r2b\_\*.
- r2i\_req[3:0]: per-bank strobes to the addressed bank; x2i\_ack[3:0]: ack routed only to the selected bank.
- i2x\_req[3:0] -> b2x\_req: bank "ready to issue" drives global command request; x2b\_ack: command-path accept from sdrc\_xfr\_ctl.
- b2x\_idle: rank FIFO empty; xfr\_ok[3:0]: one-hot selected bank; b2x\_tras\_ok: global tRAS permission used with precharge gating.
- Per-bank command FSM (sdrc\_bank\_fsm)
- b2r\_ack: capture of r2b\_\* metadata in IDLE; b2x\_req/x2b\_ack: PRE/ACT/RD/WR command request/accept.
- x2b\_pre\_ok, x2b\_act\_ok: permission gates for PRECHARGE/ACTIVATE; x2b\_rdok, x2b\_wrok: acceptance windows for READ/WRITE.
- x2b\_refresh: indicates AUTO-REFRESH activity; bank FSM re-open sequencing on refresh.
- b2x\_start/b2x\_last/b2x\_len/b2x\_wrap/b2x\_id: meta passed during IDLE and latched on accept for subsequent states.
- Transfer/controller path (sdrc\_xfr\_ctl)
- mgmt\_req/mgmt\_ack: management init/refresh arbitration (priority over backend); mgmt\_idle: no pending mgmt; sdr\_init\_done: backend enable.
- x2b\_ack: accept returned to the selected source; x2b\_rdok/x2b\_wrok: backend acceptance windows considering pipeline and mgmt.
- x2b\_act\_ok: ACTIVATE spacing; x2b\_pre\_ok[3:0]: per-bank precharge permission combined with b2x\_tras\_ok.
- x2b\_refresh: pulses during AUTO-REFRESH; rd\_pipe\_mt: read pipeline empty flag, gates write acceptance for turnaround safety.
- x2a\_wrstart/x2a\_wrnxt/x2a\_wrlast: write beat strobes; x2a\_rdstart/x2a\_rdok/x2a\_rdlst: read beat

strokes aligned to CAS latency.

- sdr\_den\_n: active-low DQ output enable; asserted low during valid write beats.
- Bus-width adapter (sdrc\_bs\_convert)
- app\_wr\_next: asserted on final narrow beat forming one 32-bit app word (gated by x2a\_wrnnext and width counters).
- app\_rd\_valid: asserted on final narrow beat assembling a 32-bit word (gated by x2a\_rdk and width counters).
- app\_last\_wr/app\_last\_rd: forwarded from x2a\_wrlast/x2a\_rdlas; realign counters at burst boundaries.
- Ordering and usage
- Request path: app req -> req\_ack (b2r\_arb\_ok) -> r2b\_req -> b2r\_ack -> b2x\_req -> x2b\_ack -> data beats via x2a\_\*.
- Read data valid: x2a\_rdk per narrow beat; app\_rd\_valid on final beat per 32-bit word.
- Write accept: x2a\_wrnnext per narrow beat; app\_wr\_next on final beat per 32-bit word.
- Command gating: always honor x2b\_rdk/x2b\_wrok/x2b\_act\_ok/x2b\_pre\_ok and b2x\_tras\_ok; allow writes only when rd\_pipe\_mt is true.
- Active-level conventions: all req/ack/ok are active-high pulses unless suffixed \*\_n; sdr\_den\_n and app byte-enables are active-low.
- Idle detection: req\_idle and b2x\_idle indicate backend idle; mgmt\_idle indicates no init/refresh activity.

-- Signal Polarity and Active Levels --

SDRAM command/control: sdr\_cmd[3:0] = {CS, RAS, CAS, WE}, all bits active-low; discrete pins sdr\_cs\_n, sdr\_ras\_n, sdr\_cas\_n, sdr\_we\_n are active-low (low asserts). Addresses (sdr\_addr, sdr\_ba) latch when CS is low. PRECHARGE A10 semantics: A10=1 selects PRECHARGE ALL (active-high for ALL), A10=0 selects bank-only.

Clock enable: sdr\_cke is active-high; held high during activity, may be driven low only in full idle for power-down.

Data I/O: sdr\_den\_n is active-low; low enables SDRAM DQ output drivers during write beats. sdr\_dqm is an active-high mask (1 masks, 0 enables); mirrors a2x\_wren\_n which is an active-low byte-enable. Application/backend handshakes and status: req, req\_ack, r2b\_req, b2r\_ack, i2x\_req, b2x\_req, x2b\_ack, x2b\_rdk, x2b\_wrok, x2b\_pre\_ok, x2b\_act\_ok, b2x\_tras\_ok are active-high. x2a\_wrstart, x2a\_wrnnext, x2a\_wrlast, x2a\_rdstart, x2a\_rdk, x2a\_rdlas are active-high strobes. x2b\_refresh pulses active-high during AUTO-REFRESH.

Write/read qualifiers and enables: req\_wr\_n is active-low (0=write, 1=read); internal r2b\_write is active-high for writes. app\_wr\_en\_n[3:0] are active-low per-byte enables; a2x\_wren\_n is an active-low narrowed enable. app\_wr\_next and app\_rd\_valid are active-high strobes.

Reset: reset\_n inputs are active-low.

-- Interface Timing Requirements --

- Clocks and CDC
- All application/handshake signals are synchronous to clk.
- Read data from pads is captured on pad\_clk and crosses into clk via a two-register path (pad\_sdr\_din -> pad\_sdr\_din1 on pad\_clk; pad\_sdr\_din1 -> pad\_sdr\_din2 on clk). Treat this as a CDC unless clk and pad\_clk are phase-related; constrain accordingly (e.g., set\_clock\_groups or modeled relationship).
- Acknowledgment pulse semantics
- req\_ack, b2r\_ack, and x2b\_ack are single-cycle pulses synchronous to clk. Sources must hold associated address/metadata/control signals stable while the request is asserted and through the ack cycle.
- Application request interface (to sdrc\_req\_gen)
- Present new requests only when b2r\_arb\_ok=1 and r2x\_idle=1.

- req, addr, len, id, rd/wr, wrap must remain stable while req=1 and until req\_ack pulses.
- After req\_ack, r2b\_req asserts for the first sub-request and must remain asserted, with its associated controls stable, until b2r\_ack. If a page split occurs, a second r2b\_req follows after the first b2r\_ack.
- Bank admission and backend handshakes (sdrc\_bank\_ctl)
  - r2b\_\* controls presented to a selected bank must remain stable until that bank asserts b2r\_ack.
  - b2x\_req asserts only when the head-of-queue bank i2x\_req is high. b2x\_cmd, b2x\_addr, b2x\_len, b2x\_id, b2x\_start, b2x\_last, b2x\_wrap must be held stable until x2b\_ack.
  - x2b\_ack is the single-cycle accept; rank FIFO advances only on READ/WRITE accept (b2x\_cmd[1]=1 and x2b\_ack=1).
- Backend command acceptance (sdrc\_xfr\_ctl)
  - Commands are accepted only when the corresponding permission is true in the same cycle: PRE when x2b\_pre\_ok & b2x\_tras\_ok, ACT when x2b\_act\_ok, READ when x2b\_rdok, WRITE when x2b\_wrok.
  - x2b\_ack indicates the command is driven on the SDRAM command bus that cycle; sources must hold b2x\_\* stable through this cycle.
  - Back-to-back ACTIVATEs are forbidden; x2b\_act\_ok enforces required spacing (with trcd/trp timers).
- SDRAM pin timing (to device)
  - sdr\_cmd is active-low {CSn,RASn,CASn,WEn}; all command/address/bank signals change synchronously to pad\_clk and must meet SDRAM AC timing.
  - sdr\_addr and sdr\_ba are latched when CS is asserted; A10=1 for PRECHARGE ALL, A10=0 for per-bank precharge.
  - sdr\_cke remains high during active operation and may be lowered only when all paths are idle per controller policy.
  - sdr\_den\_n drives low only on valid write beats; sdr\_dout and sdr\_dqm must meet setup/hold to pad\_clk when writes are accepted.
- Read data timing and CAS latency
  - x2a\_rdok (read data valid to application/width-converter) asserts cas\_latency cycles after the READ beat is scheduled (rd\_next).
  - x2a\_rdstart and x2a\_rdlas align to the same CAS pipeline; cas\_latency must match the SDRAM mode register CAS.
  - Writes are suppressed while the read-return pipeline is non-empty (rd\_pipe\_mt=0) to honor bus turnaround timing.
- Write data timing
  - x2a\_wrstart asserts on the first accepted write beat; x2a\_wrnnext asserts on each accepted beat; x2a\_wrlast asserts on the final beat.
  - During cycles with wr\_next=1, sdr\_dout and sdr\_dqm must be stable; sdr\_den\_n must be low only in those cycles.
- Bus-width conversion timing (sdrc\_bs\_convert)
  - 32-bit SDR bus: app\_wr\_next follows x2a\_wrnnext; app\_rd\_valid follows x2a\_rdok directly.
  - 16-bit SDR bus: hold app\_wr\_data/app\_wr\_en\_n across two x2a\_wrnnext pulses; app\_wr\_next asserts on the 2nd pulse. app\_rd\_valid asserts on the 2nd x2a\_rdok; sample app\_rd\_data only when app\_rd\_valid=1.
  - 8-bit SDR bus: hold app\_wr\_data/app\_wr\_en\_n across four x2a\_wrnnext pulses; app\_wr\_next asserts on the 4th pulse. app\_rd\_valid asserts on the 4th x2a\_rdok.
  - x2a\_wrlast/x2a\_rdlas mark burst boundaries; width-conversion counters reset on these pulses.

- Arbitration, refresh, and precharge timing effects
- While mgmt\_req (init/refresh) is active, x2b\_rdok/x2b\_wrok deassert; expect stalls in backend acceptance.
- Precharge permission requires cb\_pre\_ok & b2x\_tras\_ok; precharge is blocked during active streaming until both allow.
- Address/burst boundary timing
- Next READ/WRITE is emitted at sub-burst boundaries when xfr\_caddr\_lsb[1:0]==0. Choose b2x\_len and b2x\_wrap consistent with the SDRAM burst length to avoid premature BURST TERMINATE.
- Programming-dependent timing and initialization
- trp\_delay (tRP), trcd\_delay (tRCD), tRAS enforcement, trcar\_delay (tRC for auto-refresh spacing), and tMRD must be programmed to meet the SDRAM datasheet.
- No backend transfers are accepted until sdr\_init\_done=1.

## ARCHITECTURE

-- Top-Level Architecture --

sdrc\_core top-level architecture bridges a 32-bit application interface to an SDR SDRAM device with configurable data width (32/16/8). It is organized around five main blocks operating under a core/control clock (clk) and an SDRAM pad clock (pad\_clk), with a dedicated read-data CDC path. External interfaces include: application request/control (req, req\_addr, req\_len, req\_wr\_n, req\_id, req\_wrap, req\_ack, req\_idle), write data (app\_wr\_data[31:0], app\_wr\_en\_n[3:0], app\_wr\_next, app\_last\_wr), read data (app\_rd\_data[31:0], app\_rd\_valid, app\_last\_rd), and SDRAM pins (sdr\_cmd[3:0] active-low, sdr\_addr, sdr\_ba[1:0], sdr\_cke, sdr\_dqm, sdr\_dout, sdr\_den\_n active-low, sdr\_din). Configuration inputs define data width, page geometry, CAS latency, timing (tRP/tRCD/tRAS), refresh cadence, and mode register programming.

Block roles and connectivity:

- sdrc\_req\_gen (front-end): accepts application requests, normalizes length to x8-beat units per sdr\_width, maps linear address to {bank,row,column} using cfg\_col\_bits, and splits non-wrapping bursts at page boundaries. It exposes per-bank sub-requests (r2b\_\*) and handshakes admission with sdrc\_bank\_ctl via b2r\_arb\_ok/b2r\_ack.
- sdrc\_bank\_ctl + per-bank sdrc\_bank\_fsm: bank\_ctl queues head-of-line sub-requests per bank and selects one bank at a time, multiplexing command/address/meta toward the back-end (b2x\_\*), and routing acknowledgments (x2b\_ack) back to the active bank. Each bank\_fsm enforces tRP/tRCD/tRAS, tracks open-row state (page hit/miss), and sequences PRE/ACT/READ/WRITE for its bank, generating i2x\_req and related metadata. Timing guards x2b\_act\_ok/x2b\_pre\_ok and b2x\_tras\_ok gate activates and precharges.
- sdrc\_xfr\_ctl (SDRAM command/transfer + management): arbitrates between backend transfers and management (initialization/refresh); encodes/outputs SDRAM commands and pins (CS/RAS/CAS/WE, address, bank, CKE, DQM, DQ enable/data). It streams WRITE bursts, aligns READ return to cas\_latency, and provides acceptance windows x2b\_rdok/x2b\_wrok to the bank side. It performs power-up sequencing (PREALL, AUTO-REFRESH cycles, LOAD MODE) and periodic refresh, asserting sdr\_init\_done when ready, and deasserts sdr\_cke in deep idle.
- sdrc\_bs\_convert (bus-width adapter): serializes 32-bit application writes into 1/2/4 SDR beats (for

32/16/8-bit widths) with narrowed byte enables, and assembles 1/2/4 SDR beats into 32-bit application reads. It paces the application via app\_wr\_next and app\_rd\_valid only when full words are consumed-produced, and forwards last-beat indicators.

- Read-data CDC path: SDRAM DQ is captured on pad\_clk into a first stage (pad\_sdr\_din1) and transferred into the clk domain (pad\_sdr\_din2). sdrc\_xfr\_ctl schedules READ beats so that x2a\_rdok/start/last align to cas\_latency taps; writes are blocked while the read-return pipeline is non-empty to avoid turnaround hazards.

Data/control flow: application requests enter sdrc\_req\_gen, are bank-queued in sdrc\_bank\_ctl, timed and sequenced by the selected sdrc\_bank\_fsm, and executed by sdrc\_xfr\_ctl which drives SDRAM pins. Write data flows from the application through sdrc\_bs\_convert to xfr\_ctl and out to SDRAM; read data returns from SDRAM through the CDC path to xfr\_ctl and then through sdrc\_bs\_convert to the application. Handshakes and guards (b2r\_arb\_ok/b2r\_ack, x2b\_ack, x2b\_rdok/wrok, x2b\_act\_ok/x2b\_pre\_ok, b2x\_tras\_ok) ensure legal timing, bank policy, and bus turnaround behavior.

Clocking and conventions: clk is the control/application domain; pad\_clk captures SDRAM read data. The two clocks are related but require explicit CDC/timing constraints on the read path. Active-low signals include command bits (sdr\_cmd), DQ output enable (sdr\_den\_n), and byte-enable strobes; DQM polarity must match device truth tables.

Configurability and scaling: page geometry via cfg\_col\_bits, data width via sdr\_width, CAS latency and timing delays, refresh parameters, and mode register contents are parameterized. The design assumes four banks; per-bank FIFO depth in bank\_ctl may be configurable. Power-down behavior is managed centrally by sdrc\_xfr\_ctl via sdr\_cke when backend, management, and read pipelines are idle.

-- Request Generation and Address Mapping (sdrc\_req\_gen) --

Request Generation and Address Mapping (sdrc\_req\_gen)

Purpose

- Front-end request splitter and address mapper for SDRAM. Converts a generic application request into one or two page-aligned sub-requests, mapped to ROW/BANK/COL, in width-agnostic (x8-beat) units.

Interfaces (conceptual)

- Inputs: clk, reset\_n; req (valid), req\_addr, req\_len, req\_wr\_n (0=write), req\_id, req\_wrap; sdr\_width; cfg\_col\_bits; b2r\_arb\_ok; b2r\_ack.
- Outputs: req\_ack (accept), req\_idle, r2x\_idle; r2b\_req, r2b\_write, r2b\_req\_id, r2b\_ba[1:0], r2b\_raddr, r2b\_caddr, r2b\_len, r2b\_last, r2b\_wrap.

Width normalization (x8-beat baseline)

- Converts incoming length/address to x8-beat units by shifting based on SDR bus width:
- x8: shift-by-0; x16: shift-by-1; x32: shift-by-2.
- Shifts: length is scaled (len << shift); address LSBs are padded with zeros to preserve alignment.
- Assumes downstream (bank layer) operates in the same x8-beat units.

Address mapping (cfg\_col\_bits)

- Uses normalized address to derive COL/BA/ROW:
- cfg\_col\_bits: 00→8 COL bits, 01→9, 10→10, 11→11 (page sizes 256/512/1024/2048 x8-beats).
- BA: the two bits immediately above COL.
- ROW: all bits above BA.
- Exposes: r2b\_caddr (zero-extended COL to physical bus width), r2b\_ba[1:0], r2b\_raddr.

### Page-boundary split

- Computes page\_size =  $2^{\text{COL\_BITS}}$  and col\_offset from current COL.
- max\_len = page\_size - col\_offset.
- If req\_wrap=0 and req\_len > max\_len, splits into at most two sub-requests:
  - 1) First: length=max\_len to end-of-page.
  - 2) Second: remaining length starting at next page.
- If req\_wrap=1, no forced split; wrap attribute forwarded (r2b\_wrap).
- r2b\_len never crosses a page boundary. r2b\_last=1 only on the final sub-request for the original request.

### Handshakes and sequencing

- Front-end accept: accepts a new request only when b2r\_arb\_ok=1; asserts req\_ack when the request is latched and preprocessed.
- Issue/ack: asserts r2b\_req for the current segment; waits for b2r\_ack from bank layer to confirm capture.
- Sequencing: Idle → First segment → (optional) Second segment → Idle. If a second segment is required, issues it after first ack.
- Idle indication: req\_idle=1 only in Idle; r2x\_idle mirrors readiness to accept a new request.

### Forwarded attributes

- r2b\_write = ~req\_wr\_n; r2b\_req\_id = req\_id; r2b\_wrap = req\_wrap.
- r2b\_len is in x8-beat units; constrained to current page.
- At most one split per app request (maximum two sub-requests).

### Reset/initialization

- On reset\_n=0: internal state clears to Idle; req\_idle=1; r2b\_req=0; no partial request is retained.

### Assumptions/requirements

- sdr\_width must be decoded consistently system-wide. This block assumes shifts of 0/1/2 for x8/x16/x32; if a different encoding is used elsewhere (e.g., 2'b00=32-bit), a decode must translate to the correct shift amount.
- cfg\_col\_bits must match the physical SDRAM geometry; incorrect values corrupt BA/ROW/COL decoding and page calculations.
- req\_addr/req\_len definitions at the application interface must be compatible with the width normalization step so that post-shift units represent x8-beats.
- Extremely long bursts spanning multiple pages are handled as a single app request with at most one internal split here; further spanning requires additional app-level requests or reissue by upstream logic.

-- Per-Bank Queueing and Scheduler (sdrc\_bank\_ctl) --

sdrc\_bank\_ctl provides per-bank admission, ordering, and single-issue scheduling for four SDRAM banks. It decouples upstream request acceptance (r2b\_\*) from per-bank timing (in sdrc\_bank\_fsm) and from the backend transfer path (sdrc\_xfr\_ctl). Admission/backpressure: Accepts r2b\_\* requests only when a small rank FIFO has space; exposes b2r\_arb\_ok to throttle the upstream generator (sdrc\_req\_gen). Per-bank dispatch: Decodes the target bank BA[1:0] and presents the request to only that bank via r2i\_req[3:0]; aggregates per-bank accepts i2r\_ack[3:0] back to b2r\_ack (OR). Queueing: On any i2r\_ack asserted, pushes the accepted bank ID into a rank FIFO (depth parameter a2b\_req\_depth, up to 4). The FIFO stores only 2-bit BA entries, preserves arrival order across banks, supports simultaneous push/pop, and tracks occupancy via rank\_cnt; internal write-slot select (rank\_wr\_sel) derives from current depth. Flow control: Deasserts b2r\_arb\_ok when FIFO is full; guards against overflow/underflow (implementation assertions). Scheduling: Strict head-of-line policy—only the head bank in the FIFO (xfr\_ba) may issue; if its per-bank FSM is not ready (i2x\_req[head]=0), no other

bank is considered (no look-ahead). This maintains fairness by admission order and may introduce head-of-line blocking by design. Backend command/data muxing: When the head bank asserts i2x\_req, sdrc\_bank\_ctl drives b2x\_req along with b2x\_ba=xfr\_ba, b2x\_cmd/b2x\_addr and meta (id/start/last/len.wrap) multiplexed from that bank's i2x\_\* signals. Backend acknowledgment x2b\_ack is routed only to the selected bank via one-hot x2i\_ack[head]. Completion/pop: The FIFO entry is popped only when the first data-transfer command (READ/WRITE; b2x\_cmd[1]=1) for the head bank is accepted by the backend (x2b\_ack). PRECHARGE/ACTIVATE acknowledgments do not pop; the head remains until transfer begins. Status: b2x\_idle asserts when the FIFO is empty; xfr\_ok is a one-hot decode of the selected bank used to gate issue paths. Timing and gating: Per-bank SDRAM timing (tRP/tRCD/tRAS) and row state are enforced in sdrc\_bank\_fsm; sdrc\_bank\_ctl neither modifies meta nor timing. It surfaces an all-banks tRAS-OK indicator b2x\_tras\_ok (AND of per-bank flags) to the backend; x2b\_pre\_ok/x2b\_act\_ok/x2b\_rdok/x2b\_wrok are consumed within the bank FSMs. Banks are fixed at four (BA is 2 bits); ensure upstream address mapping matches. One backend command may be issued per cycle; routing is precise: only the selected bank sees backend ack, and all b2x\_\* meta follow the head bank selection.

## **OPERATION**

## **REGISTERS**

## **CONFIGURATION**