On 8/15/2013 4:04 AM, Russ Dill wrote: > On Wed, Aug 14, 2013 at 3:18 AM, Gururaja Hebbar <gururaja.hebbar@xxxxxx> wrote: >> On 8/14/2013 3:50 AM, Russ Dill wrote: >>> Changes since v1: >>> * Rebased onto new am335x PM branch >>> * Changed to use 5th param register >> [snip] [snip] >>> + >>> + wkup_m3_reset_data_pos(); >>> + if (am33xx_i2c_sleep_sequence) { >>> + pos = wkup_m3_copy_data(am33xx_i2c_sleep_sequence, >>> + i2c_sleep_sequence_sz); >> >> Why do we need to copy this data (same constant data) on every iteration? > > Because the CM3 firmware is reset after each suspend/resume cycle. The > firmware reset handler reinitializes the DMEM. Well in that why can't the i2c payload be copied to UMEM? Thanks & regards Gururaja > >>> + /* Lower 16 bits stores offset to sleep sequence */ >>> + param4 &= ~0xffff; >>> + param4 |= pos; >>> + } >>> + >>> + if (am33xx_i2c_wake_sequence) { >>> + pos = wkup_m3_copy_data(am33xx_i2c_wake_sequence, >>> + i2c_wake_sequence_sz); >>> + /* Upper 16 bits stores offset to wake sequence */ >>> + param4 &= ~0xffff0000; >>> + param4 |= pos << 16; >>> + } >>> + >> >> Seems above entire change can be done only once. >> >>> am33xx_pm->ipc.sleep_mode = IPC_CMD_DS0; >>> am33xx_pm->ipc.param1 = DS_IPC_DEFAULT; >>> am33xx_pm->ipc.param2 = DS_IPC_DEFAULT; >>> + am33xx_pm->ipc.param4 = param4; >>> >>> am33xx_pm_ipc_cmd(&am33xx_pm->ipc); >>> >>> @@ -386,6 +413,62 @@ static int __init am33xx_map_emif(void) >>> return 0; >>> } >>> >>> +static int __init am33xx_setup_sleep_sequence(void) >>> +{ >>> + int ret; >>> + int sz; >>> + const void *prop; >>> + struct device *dev; >>> + u32 freq_hz = 100000; >> >> Magic number? > > It's taken from drivers/i2c/busses/i2c-omap.c > > u32 freq = 100000; /* default to 100000 Hz */ > > I'll add a comment to that effect. > >>> + unsigned short freq_khz; >>> + >>> + /* >>> + * We put the device tree node in the I2C controller that will >>> + * be sending the sequence. i2c1 is the only controller that can >>> + * be accessed by the firmware as it is the only controller in the >>> + * WKUP domain. >> >> and on which the PMIC sits I believe? > > Yes, but this code is designed not to be PMIC specific as one could > chose to regulate VDD_CORE with any PMIC, or even with a standalone > I2C controlled regulator. > >>> + */ >>> + dev = omap_device_get_by_hwmod_name("i2c1"); >>> + if (IS_ERR(dev)) >>> + return PTR_ERR(dev); >>> + >>> + of_property_read_u32(dev->of_node, "clock-frequency", &freq_hz); >>> + freq_khz = freq_hz / 1000; >> >> Magic number? > > Nah, converting between metric prefixes this way is pretty common in the kernel. > >>> + >>> + prop = of_get_property(dev->of_node, "sleep_sequence", &sz); >>> + if (prop) { >>> + /* >>> + * Length is sequence length + 2 bytes for freq_khz, and 1 >>> + * byte for terminator. >>> + */ >>> + am33xx_i2c_sleep_sequence = kzalloc(sz + 3, GFP_KERNEL); >>> + if (!am33xx_i2c_sleep_sequence) >>> + return -ENOMEM; >>> + put_unaligned_le16(freq_khz, am33xx_i2c_sleep_sequence); >>> + memcpy(am33xx_i2c_sleep_sequence + 2, prop, sz); >> >> so, looking at entire code, it seems there is 3 memory space for same >> data (or 1 original + 2 copy) >> >> 1. in DT >> 2. in am33xx_i2c_[sleep/wake]_sequence >> 3. in SRAm by call to wkup_m3_copy_data() >> >> why not directly copy to SRAM from DT? > > As pointed out above, the firmware reset handler would wipe it out. > >>> + i2c_sleep_sequence_sz = sz + 3; >>> + } >>> + >>> + prop = of_get_property(dev->of_node, "wake_sequence", &sz); >>> + if (prop) { >>> + am33xx_i2c_wake_sequence = kzalloc(sz + 3, GFP_KERNEL); >>> + if (!am33xx_i2c_wake_sequence) { >>> + ret = -ENOMEM; >>> + goto cleanup_sleep; >>> + } >>> + put_unaligned_le16(freq_khz, am33xx_i2c_sleep_sequence); >>> + memcpy(am33xx_i2c_wake_sequence + 2, prop, sz); >>> + i2c_wake_sequence_sz = sz + 3; >>> + } >>> + >>> + return 0; >>> + >>> +cleanup_sleep: >>> + kfree(am33xx_i2c_sleep_sequence); >>> + am33xx_i2c_sleep_sequence = NULL; >>> + return ret; >>> +} >>> + >>> int __init am33xx_pm_init(void) >>> { >>> int ret; >>> @@ -451,6 +534,12 @@ int __init am33xx_pm_init(void) >>> } >>> } >>> >>> + ret = am33xx_setup_sleep_sequence(); >>> + if (ret) { >>> + pr_err("Error fetching I2C sleep/wake sequence\n"); >>> + goto err; >>> + } >>> + >>> (void) clkdm_for_each(omap_pm_clkdms_setup, NULL); >>> >>> /* CEFUSE domain can be turned off post bootup */ >>> diff --git a/arch/arm/mach-omap2/pm33xx.h b/arch/arm/mach-omap2/pm33xx.h >>> index befdd11..d0f08a5 100644 >>> --- a/arch/arm/mach-omap2/pm33xx.h >>> +++ b/arch/arm/mach-omap2/pm33xx.h >>> @@ -52,6 +52,8 @@ struct forced_standby_module { >>> }; >>> >>> int wkup_m3_copy_code(const u8 *data, size_t size); >>> +void wkup_m3_reset_data_pos(void); >>> +int wkup_m3_copy_data(const u8 *data, size_t size); >>> int wkup_m3_prepare(void); >>> void wkup_m3_register_txev_handler(void (*txev_handler)(void)); >>> >>> diff --git a/arch/arm/mach-omap2/wkup_m3.c b/arch/arm/mach-omap2/wkup_m3.c >>> index 8eaa7f3..a541de9 100644 >>> --- a/arch/arm/mach-omap2/wkup_m3.c >>> +++ b/arch/arm/mach-omap2/wkup_m3.c >>> @@ -35,6 +35,9 @@ >>> struct wkup_m3_context { >>> struct device *dev; >>> void __iomem *code; >>> + void __iomem *data; >>> + void __iomem *data_end; >>> + size_t data_size; >>> void (*txev_handler)(void); >>> }; >>> >>> @@ -50,6 +53,30 @@ int wkup_m3_copy_code(const u8 *data, size_t size) >>> return 0; >>> } >>> >>> +/* >>> + * This pair of functions allows data to be stuffed into the end of the >>> + * CM3 data memory. This is currently used for passing the I2C sleep/wake >>> + * sequences to the firmware. >>> + */ >>> + >>> +/* Clear out the pointer for data stored at the end of DMEM */ >>> +void wkup_m3_reset_data_pos(void) >>> +{ >>> + wkup_m3->data_end = wkup_m3->data + wkup_m3->data_size; >>> +} >>> + >>> +/* >>> + * Store a block of data at the end of DMEM, return the offset within DMEM >>> + * that the data is stored at, or -ENOMEM if the data did not fit >>> + */ >>> +int wkup_m3_copy_data(const u8 *data, size_t size) >>> +{ >>> + if (wkup_m3->data + size > wkup_m3->data_end) >>> + return -ENOMEM; >>> + wkup_m3->data_end -= size; >>> + memcpy_toio(wkup_m3->data_end, data, size); >>> + return wkup_m3->data_end - wkup_m3->data; >>> +} >>> >>> void wkup_m3_register_txev_handler(void (*txev_handler)(void)) >>> { >>> @@ -83,7 +110,7 @@ int wkup_m3_prepare(void) >>> static int wkup_m3_probe(struct platform_device *pdev) >>> { >>> int irq, ret = 0; >>> - struct resource *mem; >>> + struct resource *umem, *dmem; >>> >>> pm_runtime_enable(&pdev->dev); >>> >>> @@ -95,14 +122,21 @@ static int wkup_m3_probe(struct platform_device *pdev) >>> >>> irq = platform_get_irq(pdev, 0); >>> if (!irq) { >>> - dev_err(wkup_m3->dev, "no irq resource\n"); >>> + dev_err(&pdev->dev, "no irq resource\n"); >> >> unrelated change?. Better to mention this as code cleanup in commit. > > Will add a comment to that effect, the underlying error should be > fixed in the next suspend/resume patch though. > >>> + ret = -ENXIO; >>> + goto err; >>> + } >>> + >>> + umem = platform_get_resource(pdev, IORESOURCE_MEM, 0); >>> + if (!umem) { >>> + dev_err(&pdev->dev, "no UMEM resource\n"); >>> ret = -ENXIO; >>> goto err; >>> } >>> >>> - mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); >>> - if (!mem) { >>> - dev_err(wkup_m3->dev, "no memory resource\n"); >>> + dmem = platform_get_resource(pdev, IORESOURCE_MEM, 1); >>> + if (!dmem) { >>> + dev_err(&pdev->dev, "no DMEM resource\n"); >>> ret = -ENXIO; >>> goto err; >>> } >>> @@ -116,12 +150,21 @@ static int wkup_m3_probe(struct platform_device *pdev) >>> >>> wkup_m3->dev = &pdev->dev; >>> >>> - wkup_m3->code = devm_request_and_ioremap(wkup_m3->dev, mem); >>> + wkup_m3->code = devm_request_and_ioremap(wkup_m3->dev, umem); >>> + if (!wkup_m3->code) { >>> + dev_err(wkup_m3->dev, "could not ioremap UMEM\n"); >> >> why not use "pdev->dev" here? > > Either one works > >>> + ret = -EADDRNOTAVAIL; >>> + goto err; >>> + } >>> + >>> + wkup_m3->data = devm_request_and_ioremap(wkup_m3->dev, dmem); >>> if (!wkup_m3->code) { >> >> I believe this is just a copy/paste error. s/code/data > > Doh, thanks! > >>> - dev_err(wkup_m3->dev, "could not ioremap\n"); >>> + dev_err(wkup_m3->dev, "could not ioremap DMEM\n"); >> >> same as above. >> >>> ret = -EADDRNOTAVAIL; >>> goto err; >>> } >>> + wkup_m3->data_size = resource_size(dmem); >>> + wkup_m3_reset_data_pos(); >>> >>> ret = devm_request_irq(wkup_m3->dev, irq, wkup_m3_txev_handler, >>> IRQF_DISABLED, "wkup_m3_txev", NULL); >>> -- 1.8.3.2 -- To unsubscribe from this list: send the line "unsubscribe >>> linux-omap" in the body of a message to majordomo@xxxxxxxxxxxxxxx More >>> majordomo info at http://vger.kernel.org/majordomo-info.html >>> >> >> >> _______________________________________________ >> linux-arm-kernel mailing list >> linux-arm-kernel@xxxxxxxxxxxxxxxxxxx >> http://lists.infradead.org/mailman/listinfo/linux-arm-kernel -- To unsubscribe from this list: send the line "unsubscribe linux-omap" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html