verilog - SPI slave doesn't work when I follow the spec, does when I don't? -
i wrote spi slave in verilog. there implementations out there, still learning verilog , digital logic in general, , decided i'd try write on own.
my implementation works. had make change work, , making change (i think) puts implementation @ odds motorola spi spec. i'm wondering: right in thinking strange, or not understand how works?
the spi signals come in on 4 wires called sck, ss, mosi, , miso. no surprise there. fpga running @ 12mhz , spi bus running @ 1mhz. strategy sample bus spi bus on every fpga clock posedge , keep track of both current , last value sck , ss can detect edges. pass each signal through buffer reg. logic operating 2 fpga clocks behind actual events: on clock 1, latch signal in buffer; on clock 2, copy reg use logic, , on clock 3, act upon it.
i'm using spi mode 0. according spi spec, in mode 0, slave should sample mosi line on posedge of sck , transmit on negedge of sck.
so wrote way:
reg [511:0] data; // exchange 512-bit messages master. reg [2:0] sck_buffer = 0; @(posedge clock) begin // clock fpga clock sck_buffer <= {sck_buffer[1:0], sck}; end wire sck_posedge = sck_buffer[2:1] == 2'b01; wire sck_negedge = sck_buffer[2:1] == 2'b10; reg [2:0] ss_buffer = 0; @(posedge clock) begin ss_buffer <= {ss_buffer[1:0], ss}; end wire ss_posedge = ss_buffer[2:1] == 2'b01; wire ss_negedge = ss_buffer[2:1] == 2'b10; wire ss_active = ~ss_buffer[1]; reg [1:0] mosi_buffer = 0; @(posedge clock) begin mosi_buffer = {mosi_buffer[0], mosi}; end wire mosi_in = mosi_buffer[1]; assign miso = data[511]; @(posedge clock) begin if (ss_active) begin if (sck_posedge) begin // clock goes high: capture 1 bit mosi. mosi_capture <= mosi_in; end else if (sck_negedge) begin // shift captured mosi bit lsb of data , output // msb data. note: miso wire outputs data[511]. data <= {data[510:0], mosi_capture}; end end end
the code should work this:
- when ss goes active (low), data[511] output on miso virtue of wire. (no tristate here because i'm thing on bus.)
- on posedge of sck, slave samples mosi, , master gets data[511] on miso.
- on negedge of sck, slave shifts bit sampled mosi data, causing new bit shift miso output position.
when ran version, dropped bits on place. literally half of 512-bit messages sent master slave wound corrupted.
i looked @ open-source spi slave implementation , noticed didn't use negedge of sck. changed code this:
always @(posedge clock) begin if (ss_active) begin if (sck_posedge) begin // skip "capture" business , shift mosi right // data reg. data <= {data[510:0], mosi_in}; end end end
after making change, worked perfectly. totally bulletproof.
but i'm thinking, huh?
i okay. miso gets new value after posedge of sck, , it's still there on next posedge, when master samples it. wrong changing miso on negedge? though i'm running 2 fpga cycles behind spi bus due buffering, still have 12 fpga clocks every 1 tick of sck, means have 6 fpga cycles between sck negedge , next posedge. should able bit out there on miso in time, right?
in practice (in spi mode 0) update miso right after sck posedge , not worry negedge?
thanks!
i think may confused master doing. mosi gets new value after posedge of sck, , it's still there on next posedge, when master samples it. presumably mean when slave samples it, general idea reader samples on other clock edge 1 generated data, maximise setup , hold. think, in case, master should producing data on falling edge of sck, not rising edge.
in case, first thing have datasheet master device , find out it's meant doing. ideally, should scope on sclk, mosi, , ss, , find out actual master timing is. find out when master changes mosi, , mosi setup , hold sclk edge (either pos or neg) is.
your code doesn't sample mosi on sck rising edge, , second version doesn't sample on falling edge. samples somewhere near edge. problem you've got separate synchronisers on sck , mosi (and ss). generally, recipe failure, might or might not work in case, depending on specific timings. think way. if you're right, , "mosi gets new value after posedge of sck", there's long setup , short hold on mosi, , sampling fail because, time you've seen high value on sck (up 84ns after happened), mosi has changed , invalid.
the right way sample mosi (and possibly ss) on f/f clocked sck, , use small digital pll lock sck, know when read sampler output. if understand timing, , you've got huge setup , hold on you're sampling, can instead sample sck you're doing. in case, don't need synchros on mosi , ss; create enable signal sample them when they're known stable.
Comments
Post a Comment