2015-07-02 18:12 GMT+02:00 Alexander Aring <alex.aring@xxxxxxxxx>: > On Thu, Jul 02, 2015 at 05:02:00PM +0200, Baptiste Clenet wrote: >> 2015-07-01 17:16 GMT+02:00 Alexander Aring <alex.aring@xxxxxxxxx>: >> > Hi, >> > >> > On Wed, Jul 01, 2015 at 01:59:58PM +0200, Baptiste Clenet wrote: >> >> 2015-07-01 11:45 GMT+02:00 Baptiste Clenet <bapclenet@xxxxxxxxx>: >> >> > 2015-07-01 11:22 GMT+02:00 Baptiste Clenet <bapclenet@xxxxxxxxx>: >> >> >> >> >> >> 2015-07-01 10:21 GMT+02:00 Alexander Aring <alex.aring@xxxxxxxxx>: >> >> >> > Hi, >> >> >> > >> >> >> > On Tue, Jun 30, 2015 at 11:12:36AM +0200, Baptiste Clenet wrote: >> >> >> > .... >> >> >> >> >> >> >> >> root@OpenWrt:/# dmesg | grep at86rf230 >> >> >> >> [ 94.820000] at86rf230 spi32766.1: Detected at86rf212 chip version 3 >> >> >> >> [ 94.830000] at86rf230 spi32766.1: unexcept state change from 0x00 >> >> >> >> to 0x08. Actual state: 0x00 >> >> >> >> >> >> >> >> It detects the chip but yes definitely, there is problem to read the state. >> >> >> >> Will check the pins >> >> >> >> >> >> >> > >> >> >> > if you have debugfs support and mounted it, then you could dump all >> >> >> > register settings by doing something similar like: >> >> >> > >> >> >> > cat /sys/kernel/debug/regmap/spi1.0/registers >> >> >> > >> >> >> > result would be some $REGISTER <-> $VALUE mapping. >> >> >> > >> >> >> > Note: >> >> >> > >> >> >> > One interface of the 802.15.4 phy should be up for this development method, >> >> >> > because the transceiver isn't in sleep mode then. >> >> >> > >> >> >> > - Alex >> >> >> >> >> >> The mapping: >> >> >> >> >> >> root@OpenWrt:/# cat /sys/kernel/debug/regmap/spi32766.1/registers >> >> >> 01: 16 >> > >> > this looks good, because 01 is a volatile register. Means it can be >> > changed during runtime by the transceiver and regmap _always_ get the >> > current register values when we send a get request. >> > >> > And it looks good, because you don't reading zeros on this register. >> > >> >> >> 02: f6 >> >> >> 03: 10 >> >> >> 04: 20 >> >> >> 05: 60 >> >> >> 06: 80 >> >> >> 07: 2c >> >> >> 08: 25 >> >> >> 09: 77 >> >> >> 0a: 17 >> >> >> 0b: a7 >> >> >> 0c: a4 >> >> >> 0d: 01 >> >> >> 0e: 08 >> >> >> 10: 44 >> >> >> 11: a2 >> >> >> 12: f0 >> >> >> 15: 00 >> >> >> 17: 00 >> >> >> 18: 50 >> >> >> 1a: 47 >> >> >> 1b: 54 >> >> >> 1c: 07 >> >> >> 1d: 03 >> >> >> 1e: 1f >> >> >> 1f: 00 >> >> >> 20: ff >> >> >> 21: ff >> >> >> 22: ef >> >> >> 23: be >> >> >> 24: 05 >> >> >> 25: 45 >> >> >> 26: 92 >> >> >> 27: 92 >> >> >> 28: 1e >> >> >> 29: 52 >> >> >> 2a: e4 >> >> >> 2b: d0 >> >> >> 2c: 38 >> >> >> 2d: 98 >> >> >> 2e: 42 >> >> >> 2f: 53 >> >> >> >> >> >> >> >> >> -- >> >> >> Baptiste >> >> > >> >> > >> >> > >> >> > I can set a different channel and see the difference in the regmap, >> >> > I'm definitely able to communicate with the transceiver. >> > >> > Don't trust change on registers which are not volatile. Regmap will do >> > some caching after the frist GET request of a register. If the value is >> > changed, regmap will not do a get request before again the first one. >> > The cached value will be used and then we do some bit magic on the >> > cached values and send a set register value command on the bus. >> > >> > This will reduce some traffic on the bus for configuration setting only, >> > which are done by regmap. >> > >> > When you want to look if the value for channel settings is really >> > changed, then simple add the "RG_PHY_CC_CCA" value to the volatile >> > function, see [1]. Otherwise the value is cached. >> > >> > btw: >> > For transmit/receive handling we use lowlevel spi_async calls on >> > registers which are volatile. >> > >> >> > I'm not sure about my interrupt pin definition in my dts. That might >> >> > be the problem. >> >> > >> >> > in palmbus >> >> > spi >> >> > at86rf212@0 { >> >> > compatible = "atmel,at86rf212"; >> >> > reg = <1>; >> >> > interrupt-parent = <&gpio0>; >> >> > interrupts = <15 1>; >> > >> > Please use "interrupts = <15 4>;" here, 4 indicates high-level triggered >> > interrupt and I experience on some systems deadlocks (on newer driver >> > versions you will get a warning about that). >> > >> > The reason is that we need to protect some irq resources and we do a >> > disable_irq and enable_irq path. See [0]. On _some_ architectures while >> > edge-triggered while irq is disabled, the irq won't fire after >> > enable_irq. In other hand high-level triggered irq will fire after >> > enable_irq. >> > >> >> > reset-gpio = <&gpio0 16 1>; >> >> > sleep-gpio = <&gpio0 17 1>; >> >> > spi-max-frequency = <1000000>; >> > >> > Maybe try there 4-5 Mhz? >> > >> >> > }; >> >> > >> >> > >> >> > and gpio >> >> > gpio@600 { >> >> > #address-cells = <1>; >> >> > #size-cells = <0>; >> >> > interrupt-parent = <&intc>; >> >> > interrupts = <6>; >> >> > >> >> > compatible = "mtk,mt7628-gpio", "mtk,mt7621-gpio"; >> >> > reg = <0x600 0x100>; >> >> > >> >> > gpio0: bank@0 { >> >> > reg = <0>; >> >> > ... >> >> > >> >> > >> >> > I define pin 15 as the interrupt pin here but how can I check it while >> >> > OpenWRT is running? >> > >> > I can only review the at86rf212 entry, don't know how to mux your pins >> > on your architecture correctly. >> > >> > You could try to make a "cat /proc/interrupts", this will show all >> > interrupts and check if the interrupt is increased, but the interrupt >> > should only increased by receiving and transmit complete. >> > >> > ... >> >> >> >> I'm wondering how regmap works. the at86rf230 write and read functions >> >> call regmap functions. I suppose regmap communicates with spi? >> >> >> >> Are values displayed by 'cat >> >> /sys/kernel/debug/regmap/spi32766.1/registers' read from the >> >> transceiver? (updated every time) or is it in the flash of my board >> >> and change one by one when it is required? >> >> >> >> When I change the channel, it obviously changes the regmap but how may >> >> I check that at86rf230 channel is really edited? (calling spi >> >> function) >> >> >> > >> > See my above comments about "how regmap works" there are registers which >> > are cached, but also some registers which can't be cached -> volatile >> > registers. For testing you could add the RG_PHY_CC_CCA register to the >> > volatile function. >> > >> > - Alex >> > >> > [0] http://lxr.free-electrons.com/source/drivers/net/ieee802154/at86rf230.c#L930 >> > [1] http://lxr.free-electrons.com/source/drivers/net/ieee802154/at86rf230.c#L422 >> >> Hi! >> >> Thanks a lot Alex for the clarification! Regmap is clear for me now :-) >> I've looked at every single exchange over SPI for the AT86RF230, I >> better understand the functioning and how your driver initializes it. >> (I've worked ont the 212B on another platform) >> > > The transceiver is after reset inside the RESET state, look "Extended > Operation Mode" inside at86rf212 datasheet. > > On probing for init the hw we going into TRX_OFF state. > >> Everything seems fine apart from the "at86rf230 spi32766.1: unexcept >> state change from 0x00 to 0x08. Actual state: 0x00" >> 0x01 (RG_TRX_STATUS) is volatile so the value in regmap is up to date. >> When setting up the transceiver, it goes to TRX_OFF which is the >> default state after starting. Why is it unexpected? >> > > On probing we need to "wait until the state change is done" while > hw_init. This is why we introduced the SYNCED state_change function. It > blocks while it is done. See [0]. > > This synced mechanism is build above the async state change. It does: > > 1. wait_for_completion_timeout which completes until somebody called a > "complete" > 2. in the complete handler of at86rf230_async_state_change we call the > complete -> this lets wait_for_completion_timeout unblock. > > This is the big difference between the async and synced framework of > state change. > > sync -> blocks until it's done. > async -> it does not block and have a complete handler. > > In some context of linux, of course in hard/soft irqs. We can't call > synced operations because it will run some scheduling and blocks the > function. You don't want that, if you do that you will get some "BUG - > scheduling while atomic". > > We have one callback "xmit_async" which is called in such context -> > soft irq. Or the isr routine -> hard irq. > > I hope until now it's all correct what I tell you. I am also not 100% > sure of this, but this is my point of view. > >> I don't really understand all the functions related to state change >> and async ... If you've got some time to explain me that and it will >> be maybe easier for me to debug this part and see why there is an >> unexpected change. >> > > For the state change calls we don't using the regmap framework, we using > the lowlevel spi_async calls. We use volatile registers there which are > not used by the regmap framework (except dumping the register values > over debugfs). > > If regmap dumping works for you maybe then spi_async calls doesn't work > for you? We can't also use regmap for everything, we need to load the > data into the frame buffer and regmap is only for register settings, not > huge payload of data. > > > I don't know why you reading zeros only on your spi bus, I would check > if the chipselect is right when calling spi_async. Did you changed > something according the chipselect handling? Again, note when we change > the state we don't using at86rf230_write/read/_*reg functions, we use > lowlevel spi calls. > > - Alex > > [0] http://lxr.free-electrons.com/source/drivers/net/ieee802154/at86rf230.c#L749 That was a very clear and complete answer, thanks! I've gone through the process of changing the states. Effectively spi_async() has got a problem (in comparaison to regmap call). I found some problems (they're maybe only dependent on my board (spi driver)). These errors happen at hw_init, here is the process: hw_init: at86rf230_sync_state_change(lp, STATE_FORCE_TRX_OFF); at86rf230_async_state_change(lp, &lp->state, state, at86rf230_sync_state_change_complete, false); at86rf230_async_read_reg(lp, RG_TRX_STATUS, ctx, at86rf230_async_state_change_start, irq_enable); spi_async(lp->spi, &ctx->msg); (->go to 'at86rf230_async_state_change_start' on callback) at86rf230_async_state_change_start if (trx_state == ctx->to_state) go to 'at86rf230_sync_state_change_complete' else { spi_async(lp->spi, &ctx->msg); (->go to 'at86rf230_async_state_delay' on callback) - in at86rf230_async_read_reg, ctx->trx.len = 2 so the spi driver receives 0x8100 instead of 0x81 to read TRX_STATUS which results to no readings (for me)! The transceiver returns 0 ---> I set ctx->trx.len to 1 and I receive 8 (TRX_OFF) which seems good. -- in at86rf230_async_state_change_start, we check if (trx_state == ctx->to_state), current state are: trx_state 8, ctx->to_state 3, Why are we checking if ctx->to_state 3? Because it's impossible to get 3 in TRX_STATUS, isn't it? So we should check for a 8 here? This is where I am now :-) -- Baptiste -- To unsubscribe from this list: send the line "unsubscribe linux-wpan" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html