Re: [PATCH 2/3] watchdog: atcwdt200: Add driver for Andes ATCWDT200

From: Krzysztof Kozlowski

Date: Wed Jan 07 2026 - 10:46:22 EST


On 07/01/2026 15:50, CL Wang wrote:
> +
> +static int atcwdt_enable_clk(struct atcwdt_drv *drv_data)
> +{
> + struct device *dev = drv_data->wdt_dev.parent;
> + unsigned int val;
> + int ret;
> +
> + drv_data->clk = devm_clk_get(dev, NULL);

Just use API for getting enabled clock.

> + if (IS_ERR(drv_data->clk))
> + return dev_err_probe(dev, PTR_ERR(drv_data->clk),
> + "Failed to get watchdog clock\n");
> +
> + ret = clk_prepare_enable(drv_data->clk);
> + if (ret)
> + return dev_err_probe(dev, ret,
> + "Failed to enable clock\n");
> +
> + drv_data->clk_freq = clk_get_rate(drv_data->clk);
> + if (!drv_data->clk_freq)
> + return dev_err_probe(dev, -EINVAL,
> + "Failed to get clock rate\n");
> +
> + ret = device_property_read_u32(dev, "andestech,clock-source", &val);

No, read your binding. You said it is a list... list of phandles?

> + drv_data->clk_src = (!ret && val != 0) ? CTRL_CLK_SEL_PCLK : 0;
> +
> + return 0;
> +}
> +
> +static int atcwdt_init_wdt_device(struct device *dev,
> + struct atcwdt_drv *drv_data)
> +{
> + struct watchdog_device *wdd = &drv_data->wdt_dev;
> +
> + wdd->parent = dev;
> + wdd->info = &atcwdt_info;
> + wdd->ops = &atcwdt_ops;
> + wdd->timeout = ATCWDT_TIMEOUT;
> + wdd->min_timeout = 1;
> +
> + watchdog_set_nowayout(wdd, nowayout);
> + watchdog_set_drvdata(wdd, drv_data);
> +
> + return 0;
> +}
> +
> +static void atcwdt_calc_max_timeout(struct atcwdt_drv *drv_data)
> +{
> + unsigned char rst_idx = atcwdt_get_index(0xFF, TMR_RST);
> + unsigned char int_idx = atcwdt_get_index(0xFF,
> + drv_data->int_timer_type);
> +
> + drv_data->wdt_dev.max_timeout =
> + ((1U << rst_idx) + (1U << int_idx)) / drv_data->clk_freq;
> +}
> +
> +static int atcwdt_probe(struct platform_device *pdev)
> +{
> + struct device *dev = &pdev->dev;
> + struct atcwdt_drv *drv_data;
> + int ret;
> +
> + drv_data = devm_kzalloc(dev, sizeof(*drv_data), GFP_KERNEL);
> + if (!drv_data)
> + return -ENOMEM;
> +
> + platform_set_drvdata(pdev, drv_data);
> + spin_lock_init(&drv_data->lock);
> +
> + ret = atcwdt_init_wdt_device(dev, drv_data);
> + if (ret)
> + return ret;
> +
> + ret = atcwdt_init_resource(pdev, drv_data);
> + if (ret)
> + return ret;
> +
> + ret = atcwdt_enable_clk(drv_data);
> + if (ret)
> + return ret;
> +
> + ret = atcwdt_get_int_timer_type(drv_data);
> + if (ret)
> + goto disable_clk;
> +
> + atcwdt_calc_max_timeout(drv_data);
> +
> + ret = devm_watchdog_register_device(dev, &drv_data->wdt_dev);
> + if (ret)
> + goto disable_clk;
> +
> + return 0;
> +
> +disable_clk:
> + clk_disable_unprepare(drv_data->clk);
> +
> + return ret;
> +}
> +
> +static int atcwdt_suspend(struct device *dev)
> +{
> + struct atcwdt_drv *drv_data = dev_get_drvdata(dev);
> +
> + if (watchdog_active(&drv_data->wdt_dev)) {
> + atcwdt_stop(&drv_data->wdt_dev);
> + clk_disable_unprepare(drv_data->clk);
> + }
> +
> + return 0;
> +}
> +
> +static int atcwdt_resume(struct device *dev)
> +{
> + struct atcwdt_drv *drv_data = dev_get_drvdata(dev);
> + int ret = 0;
> +
> + if (watchdog_active(&drv_data->wdt_dev)) {
> + ret = clk_prepare_enable(drv_data->clk);
> + if (ret)
> + goto clk_prepare_err;

Just return.

> +
> + atcwdt_start(&drv_data->wdt_dev);
> + atcwdt_ping(&drv_data->wdt_dev);
> + }
> +
> +clk_prepare_err:
> + return ret;
> +}
> +
> +static const struct of_device_id atcwdt_match[] = {
> + { .compatible = "andestech,qilai-wdt" },

Drop, not needed.

> + { .compatible = "andestech,ae350-wdt" },
> + { /* sentinel */ },
> +};
> +MODULE_DEVICE_TABLE(of, atcwdt_match);
> +
> +static DEFINE_SIMPLE_DEV_PM_OPS(atcwdt_pm_ops, atcwdt_suspend, atcwdt_resume);
> +
> +static struct platform_driver atcwdt_driver = {
> + .probe = atcwdt_probe,
> + .driver = {
> + .name = DRV_NAME,
> + .owner = THIS_MODULE,

>From where did you get this?

> + .of_match_table = atcwdt_match,
> + .pm = pm_sleep_ptr(&atcwdt_pm_ops),
> + },
> +};
> +
> +module_platform_driver(atcwdt_driver);
> +
> +module_param(timeout, uint, 0);
> +MODULE_PARM_DESC(timeout, "Watchdog timeout in seconds (default="
> + __MODULE_STRING(ATCWDT_TIMEOUT) ")");
> +
> +module_param(nowayout, bool, 0);
> +MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default="
> + __MODULE_STRING(WATCHDOG_NOWAYOUT) ")");
> +
> +MODULE_LICENSE("GPL");
> +MODULE_AUTHOR("CL Wang <cl634@xxxxxxxxxxxxx>");
> +MODULE_DESCRIPTION("Andes ATCWDT200 Watchdog timer driver");


Best regards,
Krzysztof