2015-07-03 15:24 GMT+02:00 Baptiste Clenet <bapclenet@xxxxxxxxx>: > 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, By the way, in at86rf230_async_state_change_start, const u8 trx_state = buf[1] & TRX_STATE_MASK; but for my spi driver store the value in buf[0] instead of buf[1], I needed to change it. >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 -- 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