Received: by 2002:a25:d7c1:0:0:0:0:0 with SMTP id o184csp4213129ybg; Fri, 25 Oct 2019 15:12:32 -0700 (PDT) X-Google-Smtp-Source: APXvYqwjvk8B9QQMitW+qvOAoJJ7HKMJHQvKT/73sPU7bvVylX24854KbrtjBHf1X9rU5taBR4Qu X-Received: by 2002:a50:8f03:: with SMTP id 3mr6751818edy.195.1572041552216; Fri, 25 Oct 2019 15:12:32 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1572041552; cv=none; d=google.com; s=arc-20160816; b=RonfMffWHr30bUsnMRcHQE+73usfTVJjOT/ygQ+0C55b8pAxJ5BPaZHDTMJa8i09TX TmEqbm1a1awffLUhq4vJ6qJzaiTtajyOH34V+c6qFger3S3bS4y1Vfia7FEHoctCRqap kDv4PGIVLz/ATUtdqMvlSFaRfsUMV9Twk+9REiPXJ1JV/wUWq/XmAvi6tgoMoW7vmCuW XPQWHwsnZyO0JXj8fZvk7I0fmqvv5xBVNLDEccjfH7SOPAenHEUATXZSnPv/XN1qSTtp IYfIdqNDgv73UHxmYV1SpYUuvIenf3pkN8z2xCNKgoH7yQv4B9dkvNS5WclbMVIonVgb HqJA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :message-id:date:subject:cc:to:from:dkim-signature; bh=GfXr6ibZs7FC+cK5PjFvH2RJdymIyAIqq8NoZ14fYjU=; b=X8BFSvAtbtYi+4e/rdf3x6FivFPWU12lsFEq3N9YPmw7gR8IGQLNNnUctcQRKbQEfN fz3sO3DUz8xeOAx84AmRUJtpXl+AzwNsNon6ClM2DYtFkilyogbjXhlDdrQkFMvsBHcX MZRWFRPV7FqmJAzDqQ50sobvQ7NbUsNGbpFnknyqWqNvAPnpdKmQHNgGr9bX+TQWm4sp yDjiocNxJ/OVKuh/xN1T0AcB7QMVO0sss4+TNV3eWYJUW1icmArztEKojMjShWDFWIlx TqF2qT/AEewkg5HIdeim5kOopRpcsaoMBB9METcNbqBihapnEU/6Sq9LzaWDXTpnJ+JY Jzpw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=Pk8UuFnJ; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id q30si2593853edb.372.2019.10.25.15.11.45; Fri, 25 Oct 2019 15:12:32 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=Pk8UuFnJ; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1731585AbfJXW2Q (ORCPT + 99 others); Thu, 24 Oct 2019 18:28:16 -0400 Received: from mail-il1-f194.google.com ([209.85.166.194]:40975 "EHLO mail-il1-f194.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1731565AbfJXW2Q (ORCPT ); Thu, 24 Oct 2019 18:28:16 -0400 Received: by mail-il1-f194.google.com with SMTP id z10so92648ilo.8 for ; Thu, 24 Oct 2019 15:28:14 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=chromium.org; s=google; h=from:to:cc:subject:date:message-id:mime-version :content-transfer-encoding; bh=GfXr6ibZs7FC+cK5PjFvH2RJdymIyAIqq8NoZ14fYjU=; b=Pk8UuFnJDvmhbrsdG8ewHe6Sd4Iz8rRYkhZRaiOg+qoOaeZWIfCJAgxtdlC6pD+jjM tC28TiNjAeUX9GunDMkNdE8S9gkv4KTE48Vd8sE6S26msb4ggpKHCu12HuVe7gCdu1Ct XD35DnTgYg5uhbU71ylbqHuRx5GhYq7DonEK0= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id:mime-version :content-transfer-encoding; bh=GfXr6ibZs7FC+cK5PjFvH2RJdymIyAIqq8NoZ14fYjU=; b=TvVKtv5c/6vX5iC9cZ6U/cKLQPijK0vV8wMxzwbQE30C/Wmck1SS7mp3dt7lCC8Jop E1NroUbXwInLlifI1ySFNgbOUFRrvQwMP+lIqEisg1qlhr/XpFQkoH24LnYsnwWaZKGP 6cg4NQ4BWwnadj8+iXlmJUakpW8Fn6kZ+39c5qQJvpTsZJw4McMzpiZiHgwQh8NtqRDB 63q/w6MjjpU16DL34WTGen/5vieMoh6MaZyp8FDxQxu36LdV0NvYyDWRUXSv6IGmf74g m8CMRm6XwYeDpXv+hF3zdZwn6lwqSAh1srxLg8bu67+HqVRf1Kpdz7xv34Vl+Xgp97DR sbzg== X-Gm-Message-State: APjAAAUE5NpjPOtAFlA1r8NgniBNx9/zMQqJI1mgsIFNGWLTvZIe+a5x EDXho5S19XMGlvdYSuwFUPWNgQ== X-Received: by 2002:a92:cd41:: with SMTP id v1mr476363ilq.286.1571956094058; Thu, 24 Oct 2019 15:28:14 -0700 (PDT) Received: from ncrews2.bld.corp.google.com ([2620:15c:183:200:cb43:2cd4:65f5:5c84]) by smtp.gmail.com with ESMTPSA id x64sm39042ill.75.2019.10.24.15.28.12 (version=TLS1_2 cipher=ECDHE-RSA-AES128-SHA bits=128/128); Thu, 24 Oct 2019 15:28:13 -0700 (PDT) From: Nick Crews To: enric.balletbo@collabora.com, bleung@chromium.org, linux-leds@vger.kernel.org, jacek.anaszewski@gmail.com, pavel@ucw.cz Cc: linux-kernel@vger.kernel.org, arnd@arndb.de, weiyongjun1@huawei.com, dlaurie@chromium.org, djkurtz@chromium.org, dtor@google.com, sjg@chromium.org, groeck@chromium.com, Nick Crews , Daniel Campello Subject: [PATCH v8 1/2] platform/chrome: wilco_ec: Add keyboard backlight LED support Date: Thu, 24 Oct 2019 16:28:04 -0600 Message-Id: <20191024222805.183642-1-ncrews@chromium.org> X-Mailer: git-send-email 2.21.0 MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org The EC is in charge of controlling the keyboard backlight on the Wilco platform. We expose a standard LED class device named platform::kbd_backlight. Since the EC will never change the backlight level of its own accord, we don't need to implement a brightness_get() method. Signed-off-by: Nick Crews Signed-off-by: Daniel Campello --- v8 changes: -Removed unneeded #includes from keyboard_leds.h v7 changes: -Merged the LED stuff into the core wilco_ec module. This allows us to de-duplicate a lot of code, hide a lot of the LED internals from the core driver, and generally simplify things. -Follow reverse xmas tree variable declaration -Remove unneeded warning about BIOS not initializing the LEDs -Fix and standardize some comments and log messages. -Document all fields of wilco_keyboard_leds_msg -rm outdated and useless comment at top of core file. v6 changes: -Rebased patch -Fixed bug related to request/response buffer pointers on send_kbbl_mesg() -Now sends WILCO_KBBL_SUBCMD_SET_STATE instead of WILCO_KBBL_SUBCMD_GET_STATE command for keyboard_led_set_brightness() v5 changes: -Rename the LED device to "platform::kbd_backlight", to denote that this is the built-in system keyboard. v4 changes: -Call keyboard_led_set_brightness() directly within initialize_brightness(), instead of calling the library function. v3 changes: -Since this behaves the same as the standard Chrome OS keyboard backlight, rename the led device to "chromeos::kbd_backlight" -Move wilco_ec_keyboard_backlight_exists() into core module, so that the core does not depend upon the keyboard backlight driver. -This required moving some code into wilco-ec.h -Refactor out some common code in set_brightness() and initialize_brightness() v2 changes: -Remove and fix uses of led vs LED in kconfig -Assume BIOS initializes brightness, but double check in probe() -Remove get_brightness() callback, as EC never changes brightness by itself. -Use a __packed struct as message instead of opaque array -Add exported wilco_ec_keyboard_leds_exist() so the core driver now only creates a platform _device if relevant -Fix use of keyboard_led_set_brightness() since it can sleep BUG=b:141952530 TEST=Built and tested on a Wilco, including rmmod and modprobe of wilco-ec. drivers/platform/chrome/wilco_ec/Makefile | 3 +- drivers/platform/chrome/wilco_ec/core.c | 13 +- .../platform/chrome/wilco_ec/keyboard_leds.c | 191 ++++++++++++++++++ include/linux/platform_data/wilco-ec.h | 13 ++ 4 files changed, 215 insertions(+), 5 deletions(-) create mode 100644 drivers/platform/chrome/wilco_ec/keyboard_leds.c diff --git a/drivers/platform/chrome/wilco_ec/Makefile b/drivers/platform/chrome/wilco_ec/Makefile index bc817164596e..ecb3145cab18 100644 --- a/drivers/platform/chrome/wilco_ec/Makefile +++ b/drivers/platform/chrome/wilco_ec/Makefile @@ -1,6 +1,7 @@ # SPDX-License-Identifier: GPL-2.0 -wilco_ec-objs := core.o mailbox.o properties.o sysfs.o +wilco_ec-objs := core.o keyboard_leds.o mailbox.o \ + properties.o sysfs.o obj-$(CONFIG_WILCO_EC) += wilco_ec.o wilco_ec_debugfs-objs := debugfs.o obj-$(CONFIG_WILCO_EC_DEBUGFS) += wilco_ec_debugfs.o diff --git a/drivers/platform/chrome/wilco_ec/core.c b/drivers/platform/chrome/wilco_ec/core.c index 3724bf4b77c6..36c78e52ff3c 100644 --- a/drivers/platform/chrome/wilco_ec/core.c +++ b/drivers/platform/chrome/wilco_ec/core.c @@ -5,10 +5,6 @@ * Copyright 2018 Google LLC * * This is the entry point for the drivers that control the Wilco EC. - * This driver is responsible for several tasks: - * - Initialize the register interface that is used by wilco_ec_mailbox() - * - Create a platform device which is picked up by the debugfs driver - * - Create a platform device which is picked up by the RTC driver */ #include @@ -87,6 +83,15 @@ static int wilco_ec_probe(struct platform_device *pdev) goto unregister_debugfs; } + /* Set up the keyboard backlight LEDs. */ + ret = wilco_keyboard_leds_init(ec); + if (ret < 0) { + dev_err(dev, + "Failed to initialize keyboard LEDs: %d\n", + ret); + goto unregister_rtc; + } + ret = wilco_ec_add_sysfs(ec); if (ret < 0) { dev_err(dev, "Failed to create sysfs entries: %d", ret); diff --git a/drivers/platform/chrome/wilco_ec/keyboard_leds.c b/drivers/platform/chrome/wilco_ec/keyboard_leds.c new file mode 100644 index 000000000000..bb0edf51dfda --- /dev/null +++ b/drivers/platform/chrome/wilco_ec/keyboard_leds.c @@ -0,0 +1,191 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Keyboard backlight LED driver for the Wilco Embedded Controller + * + * Copyright 2019 Google LLC + * + * Since the EC will never change the backlight level of its own accord, + * we don't need to implement a brightness_get() method. + */ + +#include +#include +#include +#include +#include + +#define WILCO_EC_COMMAND_KBBL 0x75 +#define WILCO_KBBL_MODE_FLAG_PWM BIT(1) /* Set brightness by percent. */ +#define WILCO_KBBL_DEFAULT_BRIGHTNESS 0 + +struct wilco_keyboard_leds { + struct wilco_ec_device *ec; + struct led_classdev keyboard; +}; + +enum wilco_kbbl_subcommand { + WILCO_KBBL_SUBCMD_GET_FEATURES = 0x00, + WILCO_KBBL_SUBCMD_GET_STATE = 0x01, + WILCO_KBBL_SUBCMD_SET_STATE = 0x02, +}; + +/** + * struct wilco_keyboard_leds_msg - Message to/from EC for keyboard LED control. + * @command: Always WILCO_EC_COMMAND_KBBL. + * @status: Set by EC to 0 on success, 0xFF on failure. + * @subcmd: One of enum wilco_kbbl_subcommand. + * @reserved3: Should be 0. + * @mode: Bit flags for used mode, we want to use WILCO_KBBL_MODE_FLAG_PWM. + * @reserved5to8: Should be 0. + * @percent: Brightness in 0-100. Only meaningful in PWM mode. + * @reserved10to15: Should be 0. + */ +struct wilco_keyboard_leds_msg { + u8 command; + u8 status; + u8 subcmd; + u8 reserved3; + u8 mode; + u8 reserved5to8[4]; + u8 percent; + u8 reserved10to15[6]; +} __packed; + +/* Send a request, get a response, and check that the response is good. */ +static int send_kbbl_msg(struct wilco_ec_device *ec, + struct wilco_keyboard_leds_msg *request, + struct wilco_keyboard_leds_msg *response) +{ + struct wilco_ec_message msg; + int ret; + + memset(&msg, 0, sizeof(msg)); + msg.type = WILCO_EC_MSG_LEGACY; + msg.request_data = request; + msg.request_size = sizeof(*request); + msg.response_data = response; + msg.response_size = sizeof(*response); + + ret = wilco_ec_mailbox(ec, &msg); + if (ret < 0) { + dev_err(ec->dev, + "Failed sending keyboard LEDs command: %d", ret); + return ret; + } + + if (response->status) { + dev_err(ec->dev, + "EC reported failure sending keyboard LEDs command: %d", + response->status); + return -EIO; + } + + return 0; +} + +static int set_kbbl(struct wilco_ec_device *ec, enum led_brightness brightness) +{ + struct wilco_keyboard_leds_msg request; + struct wilco_keyboard_leds_msg response; + + memset(&request, 0, sizeof(request)); + request.command = WILCO_EC_COMMAND_KBBL; + request.subcmd = WILCO_KBBL_SUBCMD_SET_STATE; + request.mode = WILCO_KBBL_MODE_FLAG_PWM; + request.percent = brightness; + + return send_kbbl_msg(ec, &request, &response); +} + +static int kbbl_exist(struct wilco_ec_device *ec, bool *exists) +{ + struct wilco_keyboard_leds_msg request; + struct wilco_keyboard_leds_msg response; + int ret; + + memset(&request, 0, sizeof(request)); + request.command = WILCO_EC_COMMAND_KBBL; + request.subcmd = WILCO_KBBL_SUBCMD_GET_FEATURES; + + ret = send_kbbl_msg(ec, &request, &response); + if (ret < 0) + return ret; + + *exists = response.status != 0xFF; + + return 0; +} + +/** + * kbbl_init() - Initialize the state of the keyboard backlight. + * @ec: EC device to talk to. + * + * Gets the current brightness, ensuring that the BIOS already initialized the + * backlight to PWM mode. If not in PWM mode, then the current brightness is + * meaningless, so set the brightness to WILCO_KBBL_DEFAULT_BRIGHTNESS. + * + * Return: Final brightness of the keyboard, or negative error code on failure. + */ +static int kbbl_init(struct wilco_ec_device *ec) +{ + struct wilco_keyboard_leds_msg request; + struct wilco_keyboard_leds_msg response; + int ret; + + memset(&request, 0, sizeof(request)); + request.command = WILCO_EC_COMMAND_KBBL; + request.subcmd = WILCO_KBBL_SUBCMD_GET_STATE; + + ret = send_kbbl_msg(ec, &request, &response); + if (ret < 0) + return ret; + + if (response.mode & WILCO_KBBL_MODE_FLAG_PWM) + return response.percent; + + ret = set_kbbl(ec, WILCO_KBBL_DEFAULT_BRIGHTNESS); + if (ret < 0) + return ret; + + return WILCO_KBBL_DEFAULT_BRIGHTNESS; +} + +static int wilco_keyboard_leds_set(struct led_classdev *cdev, + enum led_brightness brightness) +{ + struct wilco_keyboard_leds *wkl = + container_of(cdev, struct wilco_keyboard_leds, keyboard); + return set_kbbl(wkl->ec, brightness); +} + +int wilco_keyboard_leds_init(struct wilco_ec_device *ec) +{ + struct wilco_keyboard_leds *wkl; + bool leds_exist; + int ret; + + ret = kbbl_exist(ec, &leds_exist); + if (ret < 0) { + dev_err(ec->dev, + "Failed checking keyboard LEDs support: %d", ret); + return ret; + } + if (!leds_exist) + return 0; + + wkl = devm_kzalloc(ec->dev, sizeof(*wkl), GFP_KERNEL); + if (!wkl) + return -ENOMEM; + + wkl->ec = ec; + wkl->keyboard.name = "platform::kbd_backlight"; + wkl->keyboard.max_brightness = 100; + wkl->keyboard.flags = LED_CORE_SUSPENDRESUME; + wkl->keyboard.brightness_set_blocking = wilco_keyboard_leds_set; + ret = kbbl_init(ec); + if (ret < 0) + return ret; + wkl->keyboard.brightness = ret; + + return devm_led_classdev_register(ec->dev, &wkl->keyboard); +} diff --git a/include/linux/platform_data/wilco-ec.h b/include/linux/platform_data/wilco-ec.h index ad03b586a095..0f7df3498a24 100644 --- a/include/linux/platform_data/wilco-ec.h +++ b/include/linux/platform_data/wilco-ec.h @@ -120,6 +120,19 @@ struct wilco_ec_message { */ int wilco_ec_mailbox(struct wilco_ec_device *ec, struct wilco_ec_message *msg); +/** + * wilco_keyboard_leds_init() - Set up the keyboard backlight LEDs. + * @ec: EC device to query. + * + * After this call, the keyboard backlight will be exposed through a an LED + * device at /sys/class/leds. + * + * This may sleep because it uses wilco_ec_mailbox(). + * + * Return: 0 on success, negative error code on failure. + */ +int wilco_keyboard_leds_init(struct wilco_ec_device *ec); + /* * A Property is typically a data item that is stored to NVRAM * by the EC. Each of these data items has an index associated -- 2.21.0