diff mbox series

[v2] cmd: Add a pwm command

Message ID 20201126104812.14619-1-pragnesh.patel@sifive.com
State Superseded
Delegated to: Tom Rini
Headers show
Series [v2] cmd: Add a pwm command | expand

Commit Message

Pragnesh Patel Nov. 26, 2020, 10:48 a.m. UTC
Add the command "pwm" for controlling the pwm channels. This
command provides pwm invert/config/enable/disable functionalities
via PWM uclass drivers

Signed-off-by: Pragnesh Patel <pragnesh.patel@sifive.com>
---

Changes in v2:
- Add test for pwm command


 README                    |   1 +
 cmd/Kconfig               |   6 ++
 cmd/Makefile              |   1 +
 cmd/pwm.c                 | 120 ++++++++++++++++++++++++++++++++++++++
 configs/sandbox_defconfig |   1 +
 test/cmd/Makefile         |   1 +
 test/cmd/pwm.c            |  54 +++++++++++++++++
 7 files changed, 184 insertions(+)
 create mode 100644 cmd/pwm.c
 create mode 100644 test/cmd/pwm.c

Comments

Simon Glass Nov. 30, 2020, 8:11 p.m. UTC | #1
Hi Pragnesh,

On Thu, 26 Nov 2020 at 03:48, Pragnesh Patel <pragnesh.patel@sifive.com> wrote:
>
> Add the command "pwm" for controlling the pwm channels. This
> command provides pwm invert/config/enable/disable functionalities
> via PWM uclass drivers
>
> Signed-off-by: Pragnesh Patel <pragnesh.patel@sifive.com>
> ---
>
> Changes in v2:
> - Add test for pwm command
>
>
>  README                    |   1 +
>  cmd/Kconfig               |   6 ++
>  cmd/Makefile              |   1 +
>  cmd/pwm.c                 | 120 ++++++++++++++++++++++++++++++++++++++
>  configs/sandbox_defconfig |   1 +
>  test/cmd/Makefile         |   1 +
>  test/cmd/pwm.c            |  54 +++++++++++++++++
>  7 files changed, 184 insertions(+)
>  create mode 100644 cmd/pwm.c
>  create mode 100644 test/cmd/pwm.c

Reviewed-by: Simon Glass <sjg@chromium.org>

Minor nits below

>
> diff --git a/README b/README
> index cb49aa15da..dab291e0d0 100644
> --- a/README
> +++ b/README
> @@ -3160,6 +3160,7 @@ i2c       - I2C sub-system
>  sspi   - SPI utility commands
>  base   - print or set address offset
>  printenv- print environment variables
> +pwm    - control pwm channels
>  setenv - set environment variables
>  saveenv - save environment variables to persistent storage
>  protect - enable or disable FLASH write protection
> diff --git a/cmd/Kconfig b/cmd/Kconfig
> index 1595de999b..0d085108f4 100644
> --- a/cmd/Kconfig
> +++ b/cmd/Kconfig
> @@ -918,6 +918,12 @@ config CMD_GPIO
>         help
>           GPIO support.
>
> +config CMD_PWM
> +       bool "pwm"
> +       depends on DM_PWM
> +       help
> +         Control PWM channels, this allows invert/config/enable/disable PWM channels.
> +
>  config CMD_GPT
>         bool "GPT (GUID Partition Table) command"
>         select EFI_PARTITION
> diff --git a/cmd/Makefile b/cmd/Makefile
> index dd86675bf2..75df3c136c 100644
> --- a/cmd/Makefile
> +++ b/cmd/Makefile
> @@ -120,6 +120,7 @@ endif
>  obj-$(CONFIG_CMD_PINMUX) += pinmux.o
>  obj-$(CONFIG_CMD_PMC) += pmc.o
>  obj-$(CONFIG_CMD_PSTORE) += pstore.o
> +obj-$(CONFIG_CMD_PWM) += pwm.o
>  obj-$(CONFIG_CMD_PXE) += pxe.o pxe_utils.o
>  obj-$(CONFIG_CMD_WOL) += wol.o
>  obj-$(CONFIG_CMD_QFW) += qfw.o
> diff --git a/cmd/pwm.c b/cmd/pwm.c
> new file mode 100644
> index 0000000000..f704c7a755
> --- /dev/null
> +++ b/cmd/pwm.c
> @@ -0,0 +1,120 @@
> +// SPDX-License-Identifier: GPL-2.0+
> +/*
> + * Control PWM channels
> + *
> + * Copyright (c) 2020 SiFive, Inc
> + * author: Pragnesh Patel <pragnesh.patel@sifive.com>
> + */
> +
> +#include <command.h>
> +#include <dm.h>
> +#include <pwm.h>
> +
> +enum pwm_cmd {
> +       PWM_SET_INVERT,
> +       PWM_SET_CONFIG,
> +       PWM_SET_ENABLE,
> +       PWM_SET_DISABLE,
> +};
> +
> +static int do_pwm(struct cmd_tbl *cmdtp, int flag, int argc,
> +                 char *const argv[])
> +{
> +       const char *str_cmd, *str_channel = NULL, *str_enable = NULL;
> +       const char *str_pwm = NULL, *str_period = NULL, *str_duty = NULL;
> +       enum pwm_cmd sub_cmd;
> +       struct udevice *dev;
> +       u32 channel, pwm_enable, pwm_dev, period_ns = 0, duty_ns = 0;
> +       int ret;
> +
> +       if (argc < 4)
> + show_usage:
> +               return CMD_RET_USAGE;
> +
> +       str_cmd = argv[1];
> +       argc -= 2;
> +       argv += 2;
> +
> +       if (argc > 0) {
> +               str_pwm = *argv;
> +               argc--;
> +               argv++;
> +       }
> +
> +       if (!str_pwm)
> +               goto show_usage;
> +
> +       switch (*str_cmd) {
> +       case 'i':
> +               sub_cmd = PWM_SET_INVERT;
> +               break;
> +       case 'c':
> +               sub_cmd = PWM_SET_CONFIG;
> +               break;
> +       case 'e':
> +               sub_cmd = PWM_SET_ENABLE;
> +               break;
> +       case 'd':
> +               sub_cmd = PWM_SET_DISABLE;
> +               break;
> +       default:
> +               goto show_usage;

I think it is better to use 'return CMD_RET_USAGE' at each place
rather than use a goto.

> +       }
> +
> +       if (IS_ENABLED(CONFIG_DM_PWM)) {

You don't need this because the command is only enabled if DM_PWM

> +               pwm_dev = simple_strtoul(str_pwm, NULL, 10);
> +               ret = uclass_get_device(UCLASS_PWM, pwm_dev, &dev);
> +               if (ret) {
> +                       printf("PWM: '%s' not found\n", str_pwm);

printf("pwm: ...

> +                       return cmd_process_error(cmdtp, ret);
> +               }
> +       }
> +
> +       if (argc > 0) {
> +               str_channel = *argv;
> +               channel = simple_strtoul(str_channel, NULL, 10);
> +               argc--;
> +               argv++;
> +       } else {
> +               goto show_usage;
> +       }
> +
> +       if (sub_cmd == PWM_SET_INVERT && argc > 0) {
> +               str_enable = *argv;
> +               pwm_enable = simple_strtoul(str_enable, NULL, 10);
> +               ret = pwm_set_invert(dev, channel, pwm_enable);
> +       } else if (sub_cmd == PWM_SET_CONFIG && argc == 2) {
> +               str_period = *argv;
> +               argc--;
> +               argv++;
> +               period_ns = simple_strtoul(str_period, NULL, 10);
> +
> +               if (argc > 0) {
> +                       str_duty = *argv;
> +                       duty_ns = simple_strtoul(str_duty, NULL, 10);
> +               }
> +
> +               ret = pwm_set_config(dev, channel, period_ns, duty_ns);
> +       } else if (sub_cmd == PWM_SET_ENABLE) {
> +               ret = pwm_set_enable(dev, channel, 1);
> +       } else if (sub_cmd == PWM_SET_DISABLE) {
> +               ret = pwm_set_enable(dev, channel, 0);
> +       } else {
> +               printf("PWM arguments missing\n");
> +               return CMD_RET_FAILURE;
> +       }
> +
> +       if (ret) {
> +               printf("error\n");

error (%d)

(so you print ret)

> +               return CMD_RET_FAILURE;
> +       }
> +
> +       printf("success\n");

Drop this. Success is assumed unless an error is printed

> +       return CMD_RET_SUCCESS;
> +}
> +
> +U_BOOT_CMD(pwm, 6, 0, do_pwm,
> +          "control pwm channels",
> +          "pwm <invert> <pwm_dev_num> <channel> <polarity>\n"
> +          "pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns>\n"
> +          "pwm <enable/disable> <pwm_dev_num> <channel>");

Should note that values are in decimal, since normally for U-Boot they are hex.

> diff --git a/configs/sandbox_defconfig b/configs/sandbox_defconfig
> index f2a767a4cd..7a16603461 100644
> --- a/configs/sandbox_defconfig
> +++ b/configs/sandbox_defconfig
> @@ -58,6 +58,7 @@ CONFIG_CMD_LSBLK=y
>  CONFIG_CMD_MUX=y
>  CONFIG_CMD_OSD=y
>  CONFIG_CMD_PCI=y
> +CONFIG_CMD_PWM=y
>  CONFIG_CMD_READ=y
>  CONFIG_CMD_REMOTEPROC=y
>  CONFIG_CMD_SPI=y
> diff --git a/test/cmd/Makefile b/test/cmd/Makefile
> index 859dcda239..dde98dd371 100644
> --- a/test/cmd/Makefile
> +++ b/test/cmd/Makefile
> @@ -4,3 +4,4 @@
>
>  obj-y += mem.o
>  obj-$(CONFIG_CMD_MEM_SEARCH) += mem_search.o
> +obj-$(CONFIG_CMD_PWM) += pwm.o
> diff --git a/test/cmd/pwm.c b/test/cmd/pwm.c
> new file mode 100644
> index 0000000000..987cdd1115
> --- /dev/null
> +++ b/test/cmd/pwm.c
> @@ -0,0 +1,54 @@
> +// SPDX-License-Identifier: GPL-2.0+
> +/*
> + * Test for pwm command
> + *
> + * Copyright 2020 SiFive, Inc
> + *
> + * Authors:
> + *   Pragnesh Patel <pragnesh.patel@sifive.com>
> + */
> +
> +#include <dm.h>
> +#include <dm/test.h>
> +#include <test/test.h>
> +#include <test/ut.h>
> +
> +/* Basic test of 'pwm' command */
> +static int dm_test_pwm_cmd(struct unit_test_state *uts)
> +{
> +       struct udevice *dev;
> +
> +       ut_assertok(uclass_get_device(UCLASS_PWM, 0, &dev));
> +       ut_assertnonnull(dev);
> +
> +       ut_assertok(console_record_reset_enable());
> +
> +       /* pwm <invert> <pwm_dev_num> <channel> <polarity> */
> +       run_command("pwm invert 0 0 1", 0);
> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));

Can you use ut_assert_nextline() for these?

> +       ut_asserteq_str("success", uts->actual_str);
> +
> +       run_command("pwm invert 0 0 0", 0);
> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
> +       ut_asserteq_str("success", uts->actual_str);
> +
> +       /* pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns> */
> +       run_command("pwm config 0 0 period_ns duty_ns", 0);
> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
> +       ut_asserteq_str("success", uts->actual_str);
> +
> +       /* pwm <enable/disable> <pwm_dev_num> <channel> */
> +       run_command("pwm enable 0 0", 0);
> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
> +       ut_asserteq_str("success", uts->actual_str);
> +
> +       run_command("pwm disable 0 0", 0);
> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
> +       ut_asserteq_str("success", uts->actual_str);
> +
> +       ut_assert_console_end();
> +
> +       return 0;
> +}
> +
> +DM_TEST(dm_test_pwm_cmd, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT | UT_TESTF_CONSOLE_REC);
> --
> 2.17.1
>

Regards,
Simon
Pragnesh Patel Dec. 1, 2020, 5:46 a.m. UTC | #2
Hi Simon,

>-----Original Message-----
>From: Simon Glass <sjg@chromium.org>
>Sent: 01 December 2020 01:42
>To: Pragnesh Patel <pragnesh.patel@openfive.com>
>Cc: U-Boot Mailing List <u-boot@lists.denx.de>; Atish Patra
><atish.patra@wdc.com>; Palmer Dabbelt <palmerdabbelt@google.com>; Bin
>Meng <bmeng.cn@gmail.com>; Paul Walmsley ( Sifive)
><paul.walmsley@sifive.com>; Anup Patel <anup.patel@wdc.com>; Sagar Kadam
><sagar.kadam@openfive.com>; rick <rick@andestech.com>; Naoki Hayama
><naoki.hayama@lineo.co.jp>; Marek Vasut <marek.vasut+renesas@gmail.com>;
>Patrick Delaunay <patrick.delaunay@st.com>; Adam Ford
><aford173@gmail.com>; Thomas Hebb <tommyhebb@gmail.com>; Ramon Fried
><rfried.dev@gmail.com>; Heinrich Schuchardt <xypron.glpk@gmx.de>; Bin Meng
><bin.meng@windriver.com>; Sam Protsenko <joe.skb7@gmail.com>; Miquel
>Raynal <miquel.raynal@bootlin.com>; Philippe Reynes
><philippe.reynes@softathome.com>; Frédéric Danis
><frederic.danis@collabora.com>; Patrice Chotard <patrice.chotard@st.com>;
>Vladimir Olovyannikov <vladimir.olovyannikov@broadcom.com>
>Subject: Re: [PATCH v2] cmd: Add a pwm command
>
>[External Email] Do not click links or attachments unless you recognize the
>sender and know the content is safe
>
>Hi Pragnesh,
>
>On Thu, 26 Nov 2020 at 03:48, Pragnesh Patel <pragnesh.patel@sifive.com>
>wrote:
>>
>> Add the command "pwm" for controlling the pwm channels. This command
>> provides pwm invert/config/enable/disable functionalities via PWM
>> uclass drivers
>>
>> Signed-off-by: Pragnesh Patel <pragnesh.patel@sifive.com>
>> ---
>>
>> Changes in v2:
>> - Add test for pwm command
>>
>>
>>  README                    |   1 +
>>  cmd/Kconfig               |   6 ++
>>  cmd/Makefile              |   1 +
>>  cmd/pwm.c                 | 120 ++++++++++++++++++++++++++++++++++++++
>>  configs/sandbox_defconfig |   1 +
>>  test/cmd/Makefile         |   1 +
>>  test/cmd/pwm.c            |  54 +++++++++++++++++
>>  7 files changed, 184 insertions(+)
>>  create mode 100644 cmd/pwm.c
>>  create mode 100644 test/cmd/pwm.c
>
>Reviewed-by: Simon Glass <sjg@chromium.org>
>
>Minor nits below
>
>>
>> diff --git a/README b/README
>> index cb49aa15da..dab291e0d0 100644
>> --- a/README
>> +++ b/README
>> @@ -3160,6 +3160,7 @@ i2c       - I2C sub-system
>>  sspi   - SPI utility commands
>>  base   - print or set address offset
>>  printenv- print environment variables
>> +pwm    - control pwm channels
>>  setenv - set environment variables
>>  saveenv - save environment variables to persistent storage  protect -
>> enable or disable FLASH write protection diff --git a/cmd/Kconfig
>> b/cmd/Kconfig index 1595de999b..0d085108f4 100644
>> --- a/cmd/Kconfig
>> +++ b/cmd/Kconfig
>> @@ -918,6 +918,12 @@ config CMD_GPIO
>>         help
>>           GPIO support.
>>
>> +config CMD_PWM
>> +       bool "pwm"
>> +       depends on DM_PWM
>> +       help
>> +         Control PWM channels, this allows invert/config/enable/disable PWM
>channels.
>> +
>>  config CMD_GPT
>>         bool "GPT (GUID Partition Table) command"
>>         select EFI_PARTITION
>> diff --git a/cmd/Makefile b/cmd/Makefile index dd86675bf2..75df3c136c
>> 100644
>> --- a/cmd/Makefile
>> +++ b/cmd/Makefile
>> @@ -120,6 +120,7 @@ endif
>>  obj-$(CONFIG_CMD_PINMUX) += pinmux.o
>>  obj-$(CONFIG_CMD_PMC) += pmc.o
>>  obj-$(CONFIG_CMD_PSTORE) += pstore.o
>> +obj-$(CONFIG_CMD_PWM) += pwm.o
>>  obj-$(CONFIG_CMD_PXE) += pxe.o pxe_utils.o
>>  obj-$(CONFIG_CMD_WOL) += wol.o
>>  obj-$(CONFIG_CMD_QFW) += qfw.o
>> diff --git a/cmd/pwm.c b/cmd/pwm.c
>> new file mode 100644
>> index 0000000000..f704c7a755
>> --- /dev/null
>> +++ b/cmd/pwm.c
>> @@ -0,0 +1,120 @@
>> +// SPDX-License-Identifier: GPL-2.0+
>> +/*
>> + * Control PWM channels
>> + *
>> + * Copyright (c) 2020 SiFive, Inc
>> + * author: Pragnesh Patel <pragnesh.patel@sifive.com>  */
>> +
>> +#include <command.h>
>> +#include <dm.h>
>> +#include <pwm.h>
>> +
>> +enum pwm_cmd {
>> +       PWM_SET_INVERT,
>> +       PWM_SET_CONFIG,
>> +       PWM_SET_ENABLE,
>> +       PWM_SET_DISABLE,
>> +};
>> +
>> +static int do_pwm(struct cmd_tbl *cmdtp, int flag, int argc,
>> +                 char *const argv[])
>> +{
>> +       const char *str_cmd, *str_channel = NULL, *str_enable = NULL;
>> +       const char *str_pwm = NULL, *str_period = NULL, *str_duty = NULL;
>> +       enum pwm_cmd sub_cmd;
>> +       struct udevice *dev;
>> +       u32 channel, pwm_enable, pwm_dev, period_ns = 0, duty_ns = 0;
>> +       int ret;
>> +
>> +       if (argc < 4)
>> + show_usage:
>> +               return CMD_RET_USAGE;
>> +
>> +       str_cmd = argv[1];
>> +       argc -= 2;
>> +       argv += 2;
>> +
>> +       if (argc > 0) {
>> +               str_pwm = *argv;
>> +               argc--;
>> +               argv++;
>> +       }
>> +
>> +       if (!str_pwm)
>> +               goto show_usage;
>> +
>> +       switch (*str_cmd) {
>> +       case 'i':
>> +               sub_cmd = PWM_SET_INVERT;
>> +               break;
>> +       case 'c':
>> +               sub_cmd = PWM_SET_CONFIG;
>> +               break;
>> +       case 'e':
>> +               sub_cmd = PWM_SET_ENABLE;
>> +               break;
>> +       case 'd':
>> +               sub_cmd = PWM_SET_DISABLE;
>> +               break;
>> +       default:
>> +               goto show_usage;
>
>I think it is better to use 'return CMD_RET_USAGE' at each place rather than use
>a goto.

Okay, will replace in v3.

>
>> +       }
>> +
>> +       if (IS_ENABLED(CONFIG_DM_PWM)) {
>
>You don't need this because the command is only enabled if DM_PWM

Will remove.

>
>> +               pwm_dev = simple_strtoul(str_pwm, NULL, 10);
>> +               ret = uclass_get_device(UCLASS_PWM, pwm_dev, &dev);
>> +               if (ret) {
>> +                       printf("PWM: '%s' not found\n", str_pwm);
>
>printf("pwm: ...

Will update.

>
>> +                       return cmd_process_error(cmdtp, ret);
>> +               }
>> +       }
>> +
>> +       if (argc > 0) {
>> +               str_channel = *argv;
>> +               channel = simple_strtoul(str_channel, NULL, 10);
>> +               argc--;
>> +               argv++;
>> +       } else {
>> +               goto show_usage;
>> +       }
>> +
>> +       if (sub_cmd == PWM_SET_INVERT && argc > 0) {
>> +               str_enable = *argv;
>> +               pwm_enable = simple_strtoul(str_enable, NULL, 10);
>> +               ret = pwm_set_invert(dev, channel, pwm_enable);
>> +       } else if (sub_cmd == PWM_SET_CONFIG && argc == 2) {
>> +               str_period = *argv;
>> +               argc--;
>> +               argv++;
>> +               period_ns = simple_strtoul(str_period, NULL, 10);
>> +
>> +               if (argc > 0) {
>> +                       str_duty = *argv;
>> +                       duty_ns = simple_strtoul(str_duty, NULL, 10);
>> +               }
>> +
>> +               ret = pwm_set_config(dev, channel, period_ns, duty_ns);
>> +       } else if (sub_cmd == PWM_SET_ENABLE) {
>> +               ret = pwm_set_enable(dev, channel, 1);
>> +       } else if (sub_cmd == PWM_SET_DISABLE) {
>> +               ret = pwm_set_enable(dev, channel, 0);
>> +       } else {
>> +               printf("PWM arguments missing\n");
>> +               return CMD_RET_FAILURE;
>> +       }
>> +
>> +       if (ret) {
>> +               printf("error\n");
>
>error (%d)
>
>(so you print ret)

Okay, will update.

>
>> +               return CMD_RET_FAILURE;
>> +       }
>> +
>> +       printf("success\n");
>
>Drop this. Success is assumed unless an error is printed

Okay, will drop in v3.

>
>> +       return CMD_RET_SUCCESS;
>> +}
>> +
>> +U_BOOT_CMD(pwm, 6, 0, do_pwm,
>> +          "control pwm channels",
>> +          "pwm <invert> <pwm_dev_num> <channel> <polarity>\n"
>> +          "pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns>\n"
>> +          "pwm <enable/disable> <pwm_dev_num> <channel>");
>
>Should note that values are in decimal, since normally for U-Boot they are hex.

Okay, will add a note here.

>
>> diff --git a/configs/sandbox_defconfig b/configs/sandbox_defconfig
>> index f2a767a4cd..7a16603461 100644
>> --- a/configs/sandbox_defconfig
>> +++ b/configs/sandbox_defconfig
>> @@ -58,6 +58,7 @@ CONFIG_CMD_LSBLK=y
>>  CONFIG_CMD_MUX=y
>>  CONFIG_CMD_OSD=y
>>  CONFIG_CMD_PCI=y
>> +CONFIG_CMD_PWM=y
>>  CONFIG_CMD_READ=y
>>  CONFIG_CMD_REMOTEPROC=y
>>  CONFIG_CMD_SPI=y
>> diff --git a/test/cmd/Makefile b/test/cmd/Makefile index
>> 859dcda239..dde98dd371 100644
>> --- a/test/cmd/Makefile
>> +++ b/test/cmd/Makefile
>> @@ -4,3 +4,4 @@
>>
>>  obj-y += mem.o
>>  obj-$(CONFIG_CMD_MEM_SEARCH) += mem_search.o
>> +obj-$(CONFIG_CMD_PWM) += pwm.o
>> diff --git a/test/cmd/pwm.c b/test/cmd/pwm.c new file mode 100644
>> index 0000000000..987cdd1115
>> --- /dev/null
>> +++ b/test/cmd/pwm.c
>> @@ -0,0 +1,54 @@
>> +// SPDX-License-Identifier: GPL-2.0+
>> +/*
>> + * Test for pwm command
>> + *
>> + * Copyright 2020 SiFive, Inc
>> + *
>> + * Authors:
>> + *   Pragnesh Patel <pragnesh.patel@sifive.com>
>> + */
>> +
>> +#include <dm.h>
>> +#include <dm/test.h>
>> +#include <test/test.h>
>> +#include <test/ut.h>
>> +
>> +/* Basic test of 'pwm' command */
>> +static int dm_test_pwm_cmd(struct unit_test_state *uts) {
>> +       struct udevice *dev;
>> +
>> +       ut_assertok(uclass_get_device(UCLASS_PWM, 0, &dev));
>> +       ut_assertnonnull(dev);
>> +
>> +       ut_assertok(console_record_reset_enable());
>> +
>> +       /* pwm <invert> <pwm_dev_num> <channel> <polarity> */
>> +       run_command("pwm invert 0 0 1", 0);
>> +       console_record_readline(uts->actual_str,
>> + sizeof(uts->actual_str));
>
>Can you use ut_assert_nextline() for these?

Will update in v3.

>
>> +       ut_asserteq_str("success", uts->actual_str);
>> +
>> +       run_command("pwm invert 0 0 0", 0);
>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
>> +       ut_asserteq_str("success", uts->actual_str);
>> +
>> +       /* pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns> */
>> +       run_command("pwm config 0 0 period_ns duty_ns", 0);
>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
>> +       ut_asserteq_str("success", uts->actual_str);
>> +
>> +       /* pwm <enable/disable> <pwm_dev_num> <channel> */
>> +       run_command("pwm enable 0 0", 0);
>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
>> +       ut_asserteq_str("success", uts->actual_str);
>> +
>> +       run_command("pwm disable 0 0", 0);
>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
>> +       ut_asserteq_str("success", uts->actual_str);
>> +
>> +       ut_assert_console_end();
>> +
>> +       return 0;
>> +}
>> +
>> +DM_TEST(dm_test_pwm_cmd, UT_TESTF_SCAN_PDATA |
>UT_TESTF_SCAN_FDT |
>> +UT_TESTF_CONSOLE_REC);
>> --
>> 2.17.1
>>
>
>Regards,
>Simon
Pragnesh Patel Dec. 1, 2020, 7:05 a.m. UTC | #3
Hi Simon,

>-----Original Message-----
>From: U-Boot <u-boot-bounces@lists.denx.de> On Behalf Of Pragnesh Patel
>Sent: 01 December 2020 11:17
>To: Simon Glass <sjg@chromium.org>
>Cc: U-Boot Mailing List <u-boot@lists.denx.de>; Atish Patra
><atish.patra@wdc.com>; Palmer Dabbelt <palmerdabbelt@google.com>; Bin
>Meng <bmeng.cn@gmail.com>; Paul Walmsley ( Sifive)
><paul.walmsley@sifive.com>; Anup Patel <anup.patel@wdc.com>; Sagar Kadam
><sagar.kadam@openfive.com>; rick <rick@andestech.com>; Naoki Hayama
><naoki.hayama@lineo.co.jp>; Marek Vasut <marek.vasut+renesas@gmail.com>;
>Patrick Delaunay <patrick.delaunay@st.com>; Adam Ford
><aford173@gmail.com>; Thomas Hebb <tommyhebb@gmail.com>; Ramon Fried
><rfried.dev@gmail.com>; Heinrich Schuchardt <xypron.glpk@gmx.de>; Bin Meng
><bin.meng@windriver.com>; Sam Protsenko <joe.skb7@gmail.com>; Miquel
>Raynal <miquel.raynal@bootlin.com>; Philippe Reynes
><philippe.reynes@softathome.com>; Frédéric Danis
><frederic.danis@collabora.com>; Patrice Chotard <patrice.chotard@st.com>;
>Vladimir Olovyannikov <vladimir.olovyannikov@broadcom.com>
>Subject: RE: [PATCH v2] cmd: Add a pwm command
>
>Hi Simon,
>
>>-----Original Message-----
>>From: Simon Glass <sjg@chromium.org>
>>Sent: 01 December 2020 01:42
>>To: Pragnesh Patel <pragnesh.patel@openfive.com>
>>Cc: U-Boot Mailing List <u-boot@lists.denx.de>; Atish Patra
>><atish.patra@wdc.com>; Palmer Dabbelt <palmerdabbelt@google.com>; Bin
>>Meng <bmeng.cn@gmail.com>; Paul Walmsley ( Sifive)
>><paul.walmsley@sifive.com>; Anup Patel <anup.patel@wdc.com>; Sagar
>>Kadam <sagar.kadam@openfive.com>; rick <rick@andestech.com>; Naoki
>>Hayama <naoki.hayama@lineo.co.jp>; Marek Vasut
>><marek.vasut+renesas@gmail.com>; Patrick Delaunay
>><patrick.delaunay@st.com>; Adam Ford <aford173@gmail.com>; Thomas Hebb
>><tommyhebb@gmail.com>; Ramon Fried <rfried.dev@gmail.com>; Heinrich
>>Schuchardt <xypron.glpk@gmx.de>; Bin Meng <bin.meng@windriver.com>;
>Sam
>>Protsenko <joe.skb7@gmail.com>; Miquel Raynal
>><miquel.raynal@bootlin.com>; Philippe Reynes
>><philippe.reynes@softathome.com>; Frédéric Danis
>><frederic.danis@collabora.com>; Patrice Chotard
>><patrice.chotard@st.com>; Vladimir Olovyannikov
>><vladimir.olovyannikov@broadcom.com>
>>Subject: Re: [PATCH v2] cmd: Add a pwm command
>>
>>[External Email] Do not click links or attachments unless you recognize
>>the sender and know the content is safe
>>
>>Hi Pragnesh,
>>
>>On Thu, 26 Nov 2020 at 03:48, Pragnesh Patel
>><pragnesh.patel@sifive.com>
>>wrote:
>>>
>>> Add the command "pwm" for controlling the pwm channels. This command
>>> provides pwm invert/config/enable/disable functionalities via PWM
>>> uclass drivers
>>>
>>> Signed-off-by: Pragnesh Patel <pragnesh.patel@sifive.com>
>>> ---
>>>
>>> Changes in v2:
>>> - Add test for pwm command
>>>
>>>
>>>  README                    |   1 +
>>>  cmd/Kconfig               |   6 ++
>>>  cmd/Makefile              |   1 +
>>>  cmd/pwm.c                 | 120 ++++++++++++++++++++++++++++++++++++++
>>>  configs/sandbox_defconfig |   1 +
>>>  test/cmd/Makefile         |   1 +
>>>  test/cmd/pwm.c            |  54 +++++++++++++++++
>>>  7 files changed, 184 insertions(+)
>>>  create mode 100644 cmd/pwm.c
>>>  create mode 100644 test/cmd/pwm.c
>>
>>Reviewed-by: Simon Glass <sjg@chromium.org>
>>
>>Minor nits below
>>
>>>
>>> diff --git a/README b/README
>>> index cb49aa15da..dab291e0d0 100644
>>> --- a/README
>>> +++ b/README
>>> @@ -3160,6 +3160,7 @@ i2c       - I2C sub-system
>>>  sspi   - SPI utility commands
>>>  base   - print or set address offset
>>>  printenv- print environment variables
>>> +pwm    - control pwm channels
>>>  setenv - set environment variables
>>>  saveenv - save environment variables to persistent storage  protect
>>> - enable or disable FLASH write protection diff --git a/cmd/Kconfig
>>> b/cmd/Kconfig index 1595de999b..0d085108f4 100644
>>> --- a/cmd/Kconfig
>>> +++ b/cmd/Kconfig
>>> @@ -918,6 +918,12 @@ config CMD_GPIO
>>>         help
>>>           GPIO support.
>>>
>>> +config CMD_PWM
>>> +       bool "pwm"
>>> +       depends on DM_PWM
>>> +       help
>>> +         Control PWM channels, this allows
>>> +invert/config/enable/disable PWM
>>channels.
>>> +
>>>  config CMD_GPT
>>>         bool "GPT (GUID Partition Table) command"
>>>         select EFI_PARTITION
>>> diff --git a/cmd/Makefile b/cmd/Makefile index dd86675bf2..75df3c136c
>>> 100644
>>> --- a/cmd/Makefile
>>> +++ b/cmd/Makefile
>>> @@ -120,6 +120,7 @@ endif
>>>  obj-$(CONFIG_CMD_PINMUX) += pinmux.o
>>>  obj-$(CONFIG_CMD_PMC) += pmc.o
>>>  obj-$(CONFIG_CMD_PSTORE) += pstore.o
>>> +obj-$(CONFIG_CMD_PWM) += pwm.o
>>>  obj-$(CONFIG_CMD_PXE) += pxe.o pxe_utils.o
>>>  obj-$(CONFIG_CMD_WOL) += wol.o
>>>  obj-$(CONFIG_CMD_QFW) += qfw.o
>>> diff --git a/cmd/pwm.c b/cmd/pwm.c
>>> new file mode 100644
>>> index 0000000000..f704c7a755
>>> --- /dev/null
>>> +++ b/cmd/pwm.c
>>> @@ -0,0 +1,120 @@
>>> +// SPDX-License-Identifier: GPL-2.0+
>>> +/*
>>> + * Control PWM channels
>>> + *
>>> + * Copyright (c) 2020 SiFive, Inc
>>> + * author: Pragnesh Patel <pragnesh.patel@sifive.com>  */
>>> +
>>> +#include <command.h>
>>> +#include <dm.h>
>>> +#include <pwm.h>
>>> +
>>> +enum pwm_cmd {
>>> +       PWM_SET_INVERT,
>>> +       PWM_SET_CONFIG,
>>> +       PWM_SET_ENABLE,
>>> +       PWM_SET_DISABLE,
>>> +};
>>> +
>>> +static int do_pwm(struct cmd_tbl *cmdtp, int flag, int argc,
>>> +                 char *const argv[]) {
>>> +       const char *str_cmd, *str_channel = NULL, *str_enable = NULL;
>>> +       const char *str_pwm = NULL, *str_period = NULL, *str_duty = NULL;
>>> +       enum pwm_cmd sub_cmd;
>>> +       struct udevice *dev;
>>> +       u32 channel, pwm_enable, pwm_dev, period_ns = 0, duty_ns = 0;
>>> +       int ret;
>>> +
>>> +       if (argc < 4)
>>> + show_usage:
>>> +               return CMD_RET_USAGE;
>>> +
>>> +       str_cmd = argv[1];
>>> +       argc -= 2;
>>> +       argv += 2;
>>> +
>>> +       if (argc > 0) {
>>> +               str_pwm = *argv;
>>> +               argc--;
>>> +               argv++;
>>> +       }
>>> +
>>> +       if (!str_pwm)
>>> +               goto show_usage;
>>> +
>>> +       switch (*str_cmd) {
>>> +       case 'i':
>>> +               sub_cmd = PWM_SET_INVERT;
>>> +               break;
>>> +       case 'c':
>>> +               sub_cmd = PWM_SET_CONFIG;
>>> +               break;
>>> +       case 'e':
>>> +               sub_cmd = PWM_SET_ENABLE;
>>> +               break;
>>> +       case 'd':
>>> +               sub_cmd = PWM_SET_DISABLE;
>>> +               break;
>>> +       default:
>>> +               goto show_usage;
>>
>>I think it is better to use 'return CMD_RET_USAGE' at each place rather
>>than use a goto.
>
>Okay, will replace in v3.
>
>>
>>> +       }
>>> +
>>> +       if (IS_ENABLED(CONFIG_DM_PWM)) {
>>
>>You don't need this because the command is only enabled if DM_PWM
>
>Will remove.
>
>>
>>> +               pwm_dev = simple_strtoul(str_pwm, NULL, 10);
>>> +               ret = uclass_get_device(UCLASS_PWM, pwm_dev, &dev);
>>> +               if (ret) {
>>> +                       printf("PWM: '%s' not found\n", str_pwm);
>>
>>printf("pwm: ...
>
>Will update.
>
>>
>>> +                       return cmd_process_error(cmdtp, ret);
>>> +               }
>>> +       }
>>> +
>>> +       if (argc > 0) {
>>> +               str_channel = *argv;
>>> +               channel = simple_strtoul(str_channel, NULL, 10);
>>> +               argc--;
>>> +               argv++;
>>> +       } else {
>>> +               goto show_usage;
>>> +       }
>>> +
>>> +       if (sub_cmd == PWM_SET_INVERT && argc > 0) {
>>> +               str_enable = *argv;
>>> +               pwm_enable = simple_strtoul(str_enable, NULL, 10);
>>> +               ret = pwm_set_invert(dev, channel, pwm_enable);
>>> +       } else if (sub_cmd == PWM_SET_CONFIG && argc == 2) {
>>> +               str_period = *argv;
>>> +               argc--;
>>> +               argv++;
>>> +               period_ns = simple_strtoul(str_period, NULL, 10);
>>> +
>>> +               if (argc > 0) {
>>> +                       str_duty = *argv;
>>> +                       duty_ns = simple_strtoul(str_duty, NULL, 10);
>>> +               }
>>> +
>>> +               ret = pwm_set_config(dev, channel, period_ns, duty_ns);
>>> +       } else if (sub_cmd == PWM_SET_ENABLE) {
>>> +               ret = pwm_set_enable(dev, channel, 1);
>>> +       } else if (sub_cmd == PWM_SET_DISABLE) {
>>> +               ret = pwm_set_enable(dev, channel, 0);
>>> +       } else {
>>> +               printf("PWM arguments missing\n");
>>> +               return CMD_RET_FAILURE;
>>> +       }
>>> +
>>> +       if (ret) {
>>> +               printf("error\n");
>>
>>error (%d)
>>
>>(so you print ret)
>
>Okay, will update.
>
>>
>>> +               return CMD_RET_FAILURE;
>>> +       }
>>> +
>>> +       printf("success\n");
>>
>>Drop this. Success is assumed unless an error is printed

Better to print success here because in test/cmd/pwm.c

ut_assert_nextline("");  will generate a warning like below,

In file included from test/cmd/pwm.c:14:0:
test/cmd/pwm.c: In function âdm_test_pwm_cmdâ:
test/cmd/pwm.c:28:21: warning: zero-length gnu_printf format string [-Wformat-zero-length]
  ut_assert_nextline("");
                     ^
include/test/ut.h:282:33: note: in definition of macro âut_assert_nextlineâ
  if (ut_check_console_line(uts, fmt, ##args)) {   \
                                 ^~~

Let me know if you have any other idea here.

>
>Okay, will drop in v3.
>
>>
>>> +       return CMD_RET_SUCCESS;
>>> +}
>>> +
>>> +U_BOOT_CMD(pwm, 6, 0, do_pwm,
>>> +          "control pwm channels",
>>> +          "pwm <invert> <pwm_dev_num> <channel> <polarity>\n"
>>> +          "pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns>\n"
>>> +          "pwm <enable/disable> <pwm_dev_num> <channel>");
>>
>>Should note that values are in decimal, since normally for U-Boot they are hex.
>
>Okay, will add a note here.
>
>>
>>> diff --git a/configs/sandbox_defconfig b/configs/sandbox_defconfig
>>> index f2a767a4cd..7a16603461 100644
>>> --- a/configs/sandbox_defconfig
>>> +++ b/configs/sandbox_defconfig
>>> @@ -58,6 +58,7 @@ CONFIG_CMD_LSBLK=y
>>>  CONFIG_CMD_MUX=y
>>>  CONFIG_CMD_OSD=y
>>>  CONFIG_CMD_PCI=y
>>> +CONFIG_CMD_PWM=y
>>>  CONFIG_CMD_READ=y
>>>  CONFIG_CMD_REMOTEPROC=y
>>>  CONFIG_CMD_SPI=y
>>> diff --git a/test/cmd/Makefile b/test/cmd/Makefile index
>>> 859dcda239..dde98dd371 100644
>>> --- a/test/cmd/Makefile
>>> +++ b/test/cmd/Makefile
>>> @@ -4,3 +4,4 @@
>>>
>>>  obj-y += mem.o
>>>  obj-$(CONFIG_CMD_MEM_SEARCH) += mem_search.o
>>> +obj-$(CONFIG_CMD_PWM) += pwm.o
>>> diff --git a/test/cmd/pwm.c b/test/cmd/pwm.c new file mode 100644
>>> index 0000000000..987cdd1115
>>> --- /dev/null
>>> +++ b/test/cmd/pwm.c
>>> @@ -0,0 +1,54 @@
>>> +// SPDX-License-Identifier: GPL-2.0+
>>> +/*
>>> + * Test for pwm command
>>> + *
>>> + * Copyright 2020 SiFive, Inc
>>> + *
>>> + * Authors:
>>> + *   Pragnesh Patel <pragnesh.patel@sifive.com>
>>> + */
>>> +
>>> +#include <dm.h>
>>> +#include <dm/test.h>
>>> +#include <test/test.h>
>>> +#include <test/ut.h>
>>> +
>>> +/* Basic test of 'pwm' command */
>>> +static int dm_test_pwm_cmd(struct unit_test_state *uts) {
>>> +       struct udevice *dev;
>>> +
>>> +       ut_assertok(uclass_get_device(UCLASS_PWM, 0, &dev));
>>> +       ut_assertnonnull(dev);
>>> +
>>> +       ut_assertok(console_record_reset_enable());
>>> +
>>> +       /* pwm <invert> <pwm_dev_num> <channel> <polarity> */
>>> +       run_command("pwm invert 0 0 1", 0);
>>> +       console_record_readline(uts->actual_str,
>>> + sizeof(uts->actual_str));
>>
>>Can you use ut_assert_nextline() for these?
>
>Will update in v3.
>
>>
>>> +       ut_asserteq_str("success", uts->actual_str);
>>> +
>>> +       run_command("pwm invert 0 0 0", 0);
>>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
>>> +       ut_asserteq_str("success", uts->actual_str);
>>> +
>>> +       /* pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns> */
>>> +       run_command("pwm config 0 0 period_ns duty_ns", 0);
>>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
>>> +       ut_asserteq_str("success", uts->actual_str);
>>> +
>>> +       /* pwm <enable/disable> <pwm_dev_num> <channel> */
>>> +       run_command("pwm enable 0 0", 0);
>>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
>>> +       ut_asserteq_str("success", uts->actual_str);
>>> +
>>> +       run_command("pwm disable 0 0", 0);
>>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
>>> +       ut_asserteq_str("success", uts->actual_str);
>>> +
>>> +       ut_assert_console_end();
>>> +
>>> +       return 0;
>>> +}
>>> +
>>> +DM_TEST(dm_test_pwm_cmd, UT_TESTF_SCAN_PDATA |
>>UT_TESTF_SCAN_FDT |
>>> +UT_TESTF_CONSOLE_REC);
>>> --
>>> 2.17.1
>>>
>>
>>Regards,
>>Simon
Simon Glass Dec. 3, 2020, 2:04 a.m. UTC | #4
Hi Pragnesh,

On Tue, 1 Dec 2020 at 00:05, Pragnesh Patel <pragnesh.patel@openfive.com> wrote:
>
> Hi Simon,
>
> >-----Original Message-----
> >From: U-Boot <u-boot-bounces@lists.denx.de> On Behalf Of Pragnesh Patel
> >Sent: 01 December 2020 11:17
> >To: Simon Glass <sjg@chromium.org>
> >Cc: U-Boot Mailing List <u-boot@lists.denx.de>; Atish Patra
> ><atish.patra@wdc.com>; Palmer Dabbelt <palmerdabbelt@google.com>; Bin
> >Meng <bmeng.cn@gmail.com>; Paul Walmsley ( Sifive)
> ><paul.walmsley@sifive.com>; Anup Patel <anup.patel@wdc.com>; Sagar Kadam
> ><sagar.kadam@openfive.com>; rick <rick@andestech.com>; Naoki Hayama
> ><naoki.hayama@lineo.co.jp>; Marek Vasut <marek.vasut+renesas@gmail.com>;
> >Patrick Delaunay <patrick.delaunay@st.com>; Adam Ford
> ><aford173@gmail.com>; Thomas Hebb <tommyhebb@gmail.com>; Ramon Fried
> ><rfried.dev@gmail.com>; Heinrich Schuchardt <xypron.glpk@gmx.de>; Bin Meng
> ><bin.meng@windriver.com>; Sam Protsenko <joe.skb7@gmail.com>; Miquel
> >Raynal <miquel.raynal@bootlin.com>; Philippe Reynes
> ><philippe.reynes@softathome.com>; Frédéric Danis
> ><frederic.danis@collabora.com>; Patrice Chotard <patrice.chotard@st.com>;
> >Vladimir Olovyannikov <vladimir.olovyannikov@broadcom.com>
> >Subject: RE: [PATCH v2] cmd: Add a pwm command
> >
> >Hi Simon,
> >
> >>-----Original Message-----
> >>From: Simon Glass <sjg@chromium.org>
> >>Sent: 01 December 2020 01:42
> >>To: Pragnesh Patel <pragnesh.patel@openfive.com>
> >>Cc: U-Boot Mailing List <u-boot@lists.denx.de>; Atish Patra
> >><atish.patra@wdc.com>; Palmer Dabbelt <palmerdabbelt@google.com>; Bin
> >>Meng <bmeng.cn@gmail.com>; Paul Walmsley ( Sifive)
> >><paul.walmsley@sifive.com>; Anup Patel <anup.patel@wdc.com>; Sagar
> >>Kadam <sagar.kadam@openfive.com>; rick <rick@andestech.com>; Naoki
> >>Hayama <naoki.hayama@lineo.co.jp>; Marek Vasut
> >><marek.vasut+renesas@gmail.com>; Patrick Delaunay
> >><patrick.delaunay@st.com>; Adam Ford <aford173@gmail.com>; Thomas Hebb
> >><tommyhebb@gmail.com>; Ramon Fried <rfried.dev@gmail.com>; Heinrich
> >>Schuchardt <xypron.glpk@gmx.de>; Bin Meng <bin.meng@windriver.com>;
> >Sam
> >>Protsenko <joe.skb7@gmail.com>; Miquel Raynal
> >><miquel.raynal@bootlin.com>; Philippe Reynes
> >><philippe.reynes@softathome.com>; Frédéric Danis
> >><frederic.danis@collabora.com>; Patrice Chotard
> >><patrice.chotard@st.com>; Vladimir Olovyannikov
> >><vladimir.olovyannikov@broadcom.com>
> >>Subject: Re: [PATCH v2] cmd: Add a pwm command
> >>
> >>[External Email] Do not click links or attachments unless you recognize
> >>the sender and know the content is safe
> >>
> >>Hi Pragnesh,
> >>
> >>On Thu, 26 Nov 2020 at 03:48, Pragnesh Patel
> >><pragnesh.patel@sifive.com>
> >>wrote:
> >>>
> >>> Add the command "pwm" for controlling the pwm channels. This command
> >>> provides pwm invert/config/enable/disable functionalities via PWM
> >>> uclass drivers
> >>>
> >>> Signed-off-by: Pragnesh Patel <pragnesh.patel@sifive.com>
> >>> ---
> >>>
> >>> Changes in v2:
> >>> - Add test for pwm command
> >>>
> >>>
> >>>  README                    |   1 +
> >>>  cmd/Kconfig               |   6 ++
> >>>  cmd/Makefile              |   1 +
> >>>  cmd/pwm.c                 | 120 ++++++++++++++++++++++++++++++++++++++
> >>>  configs/sandbox_defconfig |   1 +
> >>>  test/cmd/Makefile         |   1 +
> >>>  test/cmd/pwm.c            |  54 +++++++++++++++++
> >>>  7 files changed, 184 insertions(+)
> >>>  create mode 100644 cmd/pwm.c
> >>>  create mode 100644 test/cmd/pwm.c
> >>
> >>Reviewed-by: Simon Glass <sjg@chromium.org>
> >>
> >>Minor nits below
> >>
> >>>
> >>> diff --git a/README b/README
> >>> index cb49aa15da..dab291e0d0 100644
> >>> --- a/README
> >>> +++ b/README
> >>> @@ -3160,6 +3160,7 @@ i2c       - I2C sub-system
> >>>  sspi   - SPI utility commands
> >>>  base   - print or set address offset
> >>>  printenv- print environment variables
> >>> +pwm    - control pwm channels
> >>>  setenv - set environment variables
> >>>  saveenv - save environment variables to persistent storage  protect
> >>> - enable or disable FLASH write protection diff --git a/cmd/Kconfig
> >>> b/cmd/Kconfig index 1595de999b..0d085108f4 100644
> >>> --- a/cmd/Kconfig
> >>> +++ b/cmd/Kconfig
> >>> @@ -918,6 +918,12 @@ config CMD_GPIO
> >>>         help
> >>>           GPIO support.
> >>>
> >>> +config CMD_PWM
> >>> +       bool "pwm"
> >>> +       depends on DM_PWM
> >>> +       help
> >>> +         Control PWM channels, this allows
> >>> +invert/config/enable/disable PWM
> >>channels.
> >>> +
> >>>  config CMD_GPT
> >>>         bool "GPT (GUID Partition Table) command"
> >>>         select EFI_PARTITION
> >>> diff --git a/cmd/Makefile b/cmd/Makefile index dd86675bf2..75df3c136c
> >>> 100644
> >>> --- a/cmd/Makefile
> >>> +++ b/cmd/Makefile
> >>> @@ -120,6 +120,7 @@ endif
> >>>  obj-$(CONFIG_CMD_PINMUX) += pinmux.o
> >>>  obj-$(CONFIG_CMD_PMC) += pmc.o
> >>>  obj-$(CONFIG_CMD_PSTORE) += pstore.o
> >>> +obj-$(CONFIG_CMD_PWM) += pwm.o
> >>>  obj-$(CONFIG_CMD_PXE) += pxe.o pxe_utils.o
> >>>  obj-$(CONFIG_CMD_WOL) += wol.o
> >>>  obj-$(CONFIG_CMD_QFW) += qfw.o
> >>> diff --git a/cmd/pwm.c b/cmd/pwm.c
> >>> new file mode 100644
> >>> index 0000000000..f704c7a755
> >>> --- /dev/null
> >>> +++ b/cmd/pwm.c
> >>> @@ -0,0 +1,120 @@
> >>> +// SPDX-License-Identifier: GPL-2.0+
> >>> +/*
> >>> + * Control PWM channels
> >>> + *
> >>> + * Copyright (c) 2020 SiFive, Inc
> >>> + * author: Pragnesh Patel <pragnesh.patel@sifive.com>  */
> >>> +
> >>> +#include <command.h>
> >>> +#include <dm.h>
> >>> +#include <pwm.h>
> >>> +
> >>> +enum pwm_cmd {
> >>> +       PWM_SET_INVERT,
> >>> +       PWM_SET_CONFIG,
> >>> +       PWM_SET_ENABLE,
> >>> +       PWM_SET_DISABLE,
> >>> +};
> >>> +
> >>> +static int do_pwm(struct cmd_tbl *cmdtp, int flag, int argc,
> >>> +                 char *const argv[]) {
> >>> +       const char *str_cmd, *str_channel = NULL, *str_enable = NULL;
> >>> +       const char *str_pwm = NULL, *str_period = NULL, *str_duty = NULL;
> >>> +       enum pwm_cmd sub_cmd;
> >>> +       struct udevice *dev;
> >>> +       u32 channel, pwm_enable, pwm_dev, period_ns = 0, duty_ns = 0;
> >>> +       int ret;
> >>> +
> >>> +       if (argc < 4)
> >>> + show_usage:
> >>> +               return CMD_RET_USAGE;
> >>> +
> >>> +       str_cmd = argv[1];
> >>> +       argc -= 2;
> >>> +       argv += 2;
> >>> +
> >>> +       if (argc > 0) {
> >>> +               str_pwm = *argv;
> >>> +               argc--;
> >>> +               argv++;
> >>> +       }
> >>> +
> >>> +       if (!str_pwm)
> >>> +               goto show_usage;
> >>> +
> >>> +       switch (*str_cmd) {
> >>> +       case 'i':
> >>> +               sub_cmd = PWM_SET_INVERT;
> >>> +               break;
> >>> +       case 'c':
> >>> +               sub_cmd = PWM_SET_CONFIG;
> >>> +               break;
> >>> +       case 'e':
> >>> +               sub_cmd = PWM_SET_ENABLE;
> >>> +               break;
> >>> +       case 'd':
> >>> +               sub_cmd = PWM_SET_DISABLE;
> >>> +               break;
> >>> +       default:
> >>> +               goto show_usage;
> >>
> >>I think it is better to use 'return CMD_RET_USAGE' at each place rather
> >>than use a goto.
> >
> >Okay, will replace in v3.
> >
> >>
> >>> +       }
> >>> +
> >>> +       if (IS_ENABLED(CONFIG_DM_PWM)) {
> >>
> >>You don't need this because the command is only enabled if DM_PWM
> >
> >Will remove.
> >
> >>
> >>> +               pwm_dev = simple_strtoul(str_pwm, NULL, 10);
> >>> +               ret = uclass_get_device(UCLASS_PWM, pwm_dev, &dev);
> >>> +               if (ret) {
> >>> +                       printf("PWM: '%s' not found\n", str_pwm);
> >>
> >>printf("pwm: ...
> >
> >Will update.
> >
> >>
> >>> +                       return cmd_process_error(cmdtp, ret);
> >>> +               }
> >>> +       }
> >>> +
> >>> +       if (argc > 0) {
> >>> +               str_channel = *argv;
> >>> +               channel = simple_strtoul(str_channel, NULL, 10);
> >>> +               argc--;
> >>> +               argv++;
> >>> +       } else {
> >>> +               goto show_usage;
> >>> +       }
> >>> +
> >>> +       if (sub_cmd == PWM_SET_INVERT && argc > 0) {
> >>> +               str_enable = *argv;
> >>> +               pwm_enable = simple_strtoul(str_enable, NULL, 10);
> >>> +               ret = pwm_set_invert(dev, channel, pwm_enable);
> >>> +       } else if (sub_cmd == PWM_SET_CONFIG && argc == 2) {
> >>> +               str_period = *argv;
> >>> +               argc--;
> >>> +               argv++;
> >>> +               period_ns = simple_strtoul(str_period, NULL, 10);
> >>> +
> >>> +               if (argc > 0) {
> >>> +                       str_duty = *argv;
> >>> +                       duty_ns = simple_strtoul(str_duty, NULL, 10);
> >>> +               }
> >>> +
> >>> +               ret = pwm_set_config(dev, channel, period_ns, duty_ns);
> >>> +       } else if (sub_cmd == PWM_SET_ENABLE) {
> >>> +               ret = pwm_set_enable(dev, channel, 1);
> >>> +       } else if (sub_cmd == PWM_SET_DISABLE) {
> >>> +               ret = pwm_set_enable(dev, channel, 0);
> >>> +       } else {
> >>> +               printf("PWM arguments missing\n");
> >>> +               return CMD_RET_FAILURE;
> >>> +       }
> >>> +
> >>> +       if (ret) {
> >>> +               printf("error\n");
> >>
> >>error (%d)
> >>
> >>(so you print ret)
> >
> >Okay, will update.
> >
> >>
> >>> +               return CMD_RET_FAILURE;
> >>> +       }
> >>> +
> >>> +       printf("success\n");
> >>
> >>Drop this. Success is assumed unless an error is printed
>
> Better to print success here because in test/cmd/pwm.c
>
> ut_assert_nextline("");  will generate a warning like below,
>
> In file included from test/cmd/pwm.c:14:0:
> test/cmd/pwm.c: In function âdm_test_pwm_cmdâ:
> test/cmd/pwm.c:28:21: warning: zero-length gnu_printf format string [-Wformat-zero-length]
>   ut_assert_nextline("");
>                      ^
> include/test/ut.h:282:33: note: in definition of macro âut_assert_nextlineâ
>   if (ut_check_console_line(uts, fmt, ##args)) {   \
>                                  ^~~
>
> Let me know if you have any other idea here.

How about adding ut_assert_nextline_empty() ?


>
> >
> >Okay, will drop in v3.
> >
> >>
> >>> +       return CMD_RET_SUCCESS;
> >>> +}
> >>> +
> >>> +U_BOOT_CMD(pwm, 6, 0, do_pwm,
> >>> +          "control pwm channels",
> >>> +          "pwm <invert> <pwm_dev_num> <channel> <polarity>\n"
> >>> +          "pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns>\n"
> >>> +          "pwm <enable/disable> <pwm_dev_num> <channel>");
> >>
> >>Should note that values are in decimal, since normally for U-Boot they are hex.
> >
> >Okay, will add a note here.
> >
> >>
> >>> diff --git a/configs/sandbox_defconfig b/configs/sandbox_defconfig
> >>> index f2a767a4cd..7a16603461 100644
> >>> --- a/configs/sandbox_defconfig
> >>> +++ b/configs/sandbox_defconfig
> >>> @@ -58,6 +58,7 @@ CONFIG_CMD_LSBLK=y
> >>>  CONFIG_CMD_MUX=y
> >>>  CONFIG_CMD_OSD=y
> >>>  CONFIG_CMD_PCI=y
> >>> +CONFIG_CMD_PWM=y
> >>>  CONFIG_CMD_READ=y
> >>>  CONFIG_CMD_REMOTEPROC=y
> >>>  CONFIG_CMD_SPI=y
> >>> diff --git a/test/cmd/Makefile b/test/cmd/Makefile index
> >>> 859dcda239..dde98dd371 100644
> >>> --- a/test/cmd/Makefile
> >>> +++ b/test/cmd/Makefile
> >>> @@ -4,3 +4,4 @@
> >>>
> >>>  obj-y += mem.o
> >>>  obj-$(CONFIG_CMD_MEM_SEARCH) += mem_search.o
> >>> +obj-$(CONFIG_CMD_PWM) += pwm.o
> >>> diff --git a/test/cmd/pwm.c b/test/cmd/pwm.c new file mode 100644
> >>> index 0000000000..987cdd1115
> >>> --- /dev/null
> >>> +++ b/test/cmd/pwm.c
> >>> @@ -0,0 +1,54 @@
> >>> +// SPDX-License-Identifier: GPL-2.0+
> >>> +/*
> >>> + * Test for pwm command
> >>> + *
> >>> + * Copyright 2020 SiFive, Inc
> >>> + *
> >>> + * Authors:
> >>> + *   Pragnesh Patel <pragnesh.patel@sifive.com>
> >>> + */
> >>> +
> >>> +#include <dm.h>
> >>> +#include <dm/test.h>
> >>> +#include <test/test.h>
> >>> +#include <test/ut.h>
> >>> +
> >>> +/* Basic test of 'pwm' command */
> >>> +static int dm_test_pwm_cmd(struct unit_test_state *uts) {
> >>> +       struct udevice *dev;
> >>> +
> >>> +       ut_assertok(uclass_get_device(UCLASS_PWM, 0, &dev));
> >>> +       ut_assertnonnull(dev);
> >>> +
> >>> +       ut_assertok(console_record_reset_enable());
> >>> +
> >>> +       /* pwm <invert> <pwm_dev_num> <channel> <polarity> */
> >>> +       run_command("pwm invert 0 0 1", 0);
> >>> +       console_record_readline(uts->actual_str,
> >>> + sizeof(uts->actual_str));
> >>
> >>Can you use ut_assert_nextline() for these?
> >
> >Will update in v3.
> >
> >>
> >>> +       ut_asserteq_str("success", uts->actual_str);
> >>> +
> >>> +       run_command("pwm invert 0 0 0", 0);
> >>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
> >>> +       ut_asserteq_str("success", uts->actual_str);
> >>> +
> >>> +       /* pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns> */
> >>> +       run_command("pwm config 0 0 period_ns duty_ns", 0);
> >>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
> >>> +       ut_asserteq_str("success", uts->actual_str);
> >>> +
> >>> +       /* pwm <enable/disable> <pwm_dev_num> <channel> */
> >>> +       run_command("pwm enable 0 0", 0);
> >>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
> >>> +       ut_asserteq_str("success", uts->actual_str);
> >>> +
> >>> +       run_command("pwm disable 0 0", 0);
> >>> +       console_record_readline(uts->actual_str, sizeof(uts->actual_str));
> >>> +       ut_asserteq_str("success", uts->actual_str);
> >>> +
> >>> +       ut_assert_console_end();
> >>> +
> >>> +       return 0;
> >>> +}
> >>> +
> >>> +DM_TEST(dm_test_pwm_cmd, UT_TESTF_SCAN_PDATA |
> >>UT_TESTF_SCAN_FDT |
> >>> +UT_TESTF_CONSOLE_REC);
> >>> --
> >>> 2.17.1
> >>>
> >>

Regards,
Simon
diff mbox series

Patch

diff --git a/README b/README
index cb49aa15da..dab291e0d0 100644
--- a/README
+++ b/README
@@ -3160,6 +3160,7 @@  i2c	- I2C sub-system
 sspi	- SPI utility commands
 base	- print or set address offset
 printenv- print environment variables
+pwm	- control pwm channels
 setenv	- set environment variables
 saveenv - save environment variables to persistent storage
 protect - enable or disable FLASH write protection
diff --git a/cmd/Kconfig b/cmd/Kconfig
index 1595de999b..0d085108f4 100644
--- a/cmd/Kconfig
+++ b/cmd/Kconfig
@@ -918,6 +918,12 @@  config CMD_GPIO
 	help
 	  GPIO support.
 
+config CMD_PWM
+	bool "pwm"
+	depends on DM_PWM
+	help
+	  Control PWM channels, this allows invert/config/enable/disable PWM channels.
+
 config CMD_GPT
 	bool "GPT (GUID Partition Table) command"
 	select EFI_PARTITION
diff --git a/cmd/Makefile b/cmd/Makefile
index dd86675bf2..75df3c136c 100644
--- a/cmd/Makefile
+++ b/cmd/Makefile
@@ -120,6 +120,7 @@  endif
 obj-$(CONFIG_CMD_PINMUX) += pinmux.o
 obj-$(CONFIG_CMD_PMC) += pmc.o
 obj-$(CONFIG_CMD_PSTORE) += pstore.o
+obj-$(CONFIG_CMD_PWM) += pwm.o
 obj-$(CONFIG_CMD_PXE) += pxe.o pxe_utils.o
 obj-$(CONFIG_CMD_WOL) += wol.o
 obj-$(CONFIG_CMD_QFW) += qfw.o
diff --git a/cmd/pwm.c b/cmd/pwm.c
new file mode 100644
index 0000000000..f704c7a755
--- /dev/null
+++ b/cmd/pwm.c
@@ -0,0 +1,120 @@ 
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Control PWM channels
+ *
+ * Copyright (c) 2020 SiFive, Inc
+ * author: Pragnesh Patel <pragnesh.patel@sifive.com>
+ */
+
+#include <command.h>
+#include <dm.h>
+#include <pwm.h>
+
+enum pwm_cmd {
+	PWM_SET_INVERT,
+	PWM_SET_CONFIG,
+	PWM_SET_ENABLE,
+	PWM_SET_DISABLE,
+};
+
+static int do_pwm(struct cmd_tbl *cmdtp, int flag, int argc,
+		  char *const argv[])
+{
+	const char *str_cmd, *str_channel = NULL, *str_enable = NULL;
+	const char *str_pwm = NULL, *str_period = NULL, *str_duty = NULL;
+	enum pwm_cmd sub_cmd;
+	struct udevice *dev;
+	u32 channel, pwm_enable, pwm_dev, period_ns = 0, duty_ns = 0;
+	int ret;
+
+	if (argc < 4)
+ show_usage:
+		return CMD_RET_USAGE;
+
+	str_cmd = argv[1];
+	argc -= 2;
+	argv += 2;
+
+	if (argc > 0) {
+		str_pwm = *argv;
+		argc--;
+		argv++;
+	}
+
+	if (!str_pwm)
+		goto show_usage;
+
+	switch (*str_cmd) {
+	case 'i':
+		sub_cmd = PWM_SET_INVERT;
+		break;
+	case 'c':
+		sub_cmd = PWM_SET_CONFIG;
+		break;
+	case 'e':
+		sub_cmd = PWM_SET_ENABLE;
+		break;
+	case 'd':
+		sub_cmd = PWM_SET_DISABLE;
+		break;
+	default:
+		goto show_usage;
+	}
+
+	if (IS_ENABLED(CONFIG_DM_PWM)) {
+		pwm_dev = simple_strtoul(str_pwm, NULL, 10);
+		ret = uclass_get_device(UCLASS_PWM, pwm_dev, &dev);
+		if (ret) {
+			printf("PWM: '%s' not found\n", str_pwm);
+			return cmd_process_error(cmdtp, ret);
+		}
+	}
+
+	if (argc > 0) {
+		str_channel = *argv;
+		channel = simple_strtoul(str_channel, NULL, 10);
+		argc--;
+		argv++;
+	} else {
+		goto show_usage;
+	}
+
+	if (sub_cmd == PWM_SET_INVERT && argc > 0) {
+		str_enable = *argv;
+		pwm_enable = simple_strtoul(str_enable, NULL, 10);
+		ret = pwm_set_invert(dev, channel, pwm_enable);
+	} else if (sub_cmd == PWM_SET_CONFIG && argc == 2) {
+		str_period = *argv;
+		argc--;
+		argv++;
+		period_ns = simple_strtoul(str_period, NULL, 10);
+
+		if (argc > 0) {
+			str_duty = *argv;
+			duty_ns = simple_strtoul(str_duty, NULL, 10);
+		}
+
+		ret = pwm_set_config(dev, channel, period_ns, duty_ns);
+	} else if (sub_cmd == PWM_SET_ENABLE) {
+		ret = pwm_set_enable(dev, channel, 1);
+	} else if (sub_cmd == PWM_SET_DISABLE) {
+		ret = pwm_set_enable(dev, channel, 0);
+	} else {
+		printf("PWM arguments missing\n");
+		return CMD_RET_FAILURE;
+	}
+
+	if (ret) {
+		printf("error\n");
+		return CMD_RET_FAILURE;
+	}
+
+	printf("success\n");
+	return CMD_RET_SUCCESS;
+}
+
+U_BOOT_CMD(pwm, 6, 0, do_pwm,
+	   "control pwm channels",
+	   "pwm <invert> <pwm_dev_num> <channel> <polarity>\n"
+	   "pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns>\n"
+	   "pwm <enable/disable> <pwm_dev_num> <channel>");
diff --git a/configs/sandbox_defconfig b/configs/sandbox_defconfig
index f2a767a4cd..7a16603461 100644
--- a/configs/sandbox_defconfig
+++ b/configs/sandbox_defconfig
@@ -58,6 +58,7 @@  CONFIG_CMD_LSBLK=y
 CONFIG_CMD_MUX=y
 CONFIG_CMD_OSD=y
 CONFIG_CMD_PCI=y
+CONFIG_CMD_PWM=y
 CONFIG_CMD_READ=y
 CONFIG_CMD_REMOTEPROC=y
 CONFIG_CMD_SPI=y
diff --git a/test/cmd/Makefile b/test/cmd/Makefile
index 859dcda239..dde98dd371 100644
--- a/test/cmd/Makefile
+++ b/test/cmd/Makefile
@@ -4,3 +4,4 @@ 
 
 obj-y += mem.o
 obj-$(CONFIG_CMD_MEM_SEARCH) += mem_search.o
+obj-$(CONFIG_CMD_PWM) += pwm.o
diff --git a/test/cmd/pwm.c b/test/cmd/pwm.c
new file mode 100644
index 0000000000..987cdd1115
--- /dev/null
+++ b/test/cmd/pwm.c
@@ -0,0 +1,54 @@ 
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Test for pwm command
+ *
+ * Copyright 2020 SiFive, Inc
+ *
+ * Authors:
+ *   Pragnesh Patel <pragnesh.patel@sifive.com>
+ */
+
+#include <dm.h>
+#include <dm/test.h>
+#include <test/test.h>
+#include <test/ut.h>
+
+/* Basic test of 'pwm' command */
+static int dm_test_pwm_cmd(struct unit_test_state *uts)
+{
+	struct udevice *dev;
+
+	ut_assertok(uclass_get_device(UCLASS_PWM, 0, &dev));
+	ut_assertnonnull(dev);
+
+	ut_assertok(console_record_reset_enable());
+
+	/* pwm <invert> <pwm_dev_num> <channel> <polarity> */
+	run_command("pwm invert 0 0 1", 0);
+	console_record_readline(uts->actual_str, sizeof(uts->actual_str));
+	ut_asserteq_str("success", uts->actual_str);
+
+	run_command("pwm invert 0 0 0", 0);
+	console_record_readline(uts->actual_str, sizeof(uts->actual_str));
+	ut_asserteq_str("success", uts->actual_str);
+
+	/* pwm <config> <pwm_dev_num> <channel> <period_ns> <duty_ns> */
+	run_command("pwm config 0 0 period_ns duty_ns", 0);
+	console_record_readline(uts->actual_str, sizeof(uts->actual_str));
+	ut_asserteq_str("success", uts->actual_str);
+
+	/* pwm <enable/disable> <pwm_dev_num> <channel> */
+	run_command("pwm enable 0 0", 0);
+	console_record_readline(uts->actual_str, sizeof(uts->actual_str));
+	ut_asserteq_str("success", uts->actual_str);
+
+	run_command("pwm disable 0 0", 0);
+	console_record_readline(uts->actual_str, sizeof(uts->actual_str));
+	ut_asserteq_str("success", uts->actual_str);
+
+	ut_assert_console_end();
+
+	return 0;
+}
+
+DM_TEST(dm_test_pwm_cmd, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT | UT_TESTF_CONSOLE_REC);