czw., 16 sty 2025 o 19:17 Pengyu Luo <mitltlatltl@xxxxxxxxx> napisał(a): > > On Fri, Jan 17, 2025 at 1:31 AM Bryan O'Donoghue <bryan.odonoghue@xxxxxxxxxx> wrote: > > On 16/01/2025 11:15, Pengyu Luo wrote: > > > There are three variants of which Huawei released the first two > > > simultaneously. > > > > > > Huawei Matebook E Go LTE(sc8180x), codename seems to be gaokun2. > > > Huawei Matebook E Go(sc8280xp@3.0GHz), codename must be gaokun3. (see [1]) > > > Huawei Matebook E Go 2023(sc8280xp@2.69GHz), codename should be also gaokun3. > > > > > > Adding support for the latter two variants for now, this driver should > > > also work for the sc8180x variant according to acpi table files, but I > > > don't have the device to test yet. > > > > > > Different from other Qualcomm Snapdragon sc8280xp based machines, the > > > Huawei Matebook E Go uses an embedded controller while others use > > > a system called PMIC GLink. This embedded controller can be used to > > > perform a set of various functions, including, but not limited to: > > > > > > - Battery and charger monitoring; > > > - Charge control and smart charge; > > > - Fn_lock settings; > > > - Tablet lid status; > > > - Temperature sensors; > > > - USB Type-C notifications (ports orientation, DP alt mode HPD); > > > - USB Type-C PD (according to observation, up to 48w). > > > > > > Add a driver for the EC which creates devices for UCSI and power supply > > > devices. > > > > > > This driver is inspired by the following drivers: > > > drivers/platform/arm64/acer-aspire1-ec.c > > > drivers/platform/arm64/lenovo-yoga-c630.c > > > drivers/platform/x86/huawei-wmi.c > > > > > > [1] https://bugzilla.kernel.org/show_bug.cgi?id=219645 > > > > > > Signed-off-by: Pengyu Luo <mitltlatltl@xxxxxxxxx> > > > --- > > > MAINTAINERS | 7 + > > > drivers/platform/arm64/Kconfig | 21 + > > > drivers/platform/arm64/Makefile | 1 + > > > drivers/platform/arm64/huawei-gaokun-ec.c | 787 ++++++++++++++++++ > > > .../linux/platform_data/huawei-gaokun-ec.h | 80 ++ > > > 5 files changed, 896 insertions(+) > > > create mode 100644 drivers/platform/arm64/huawei-gaokun-ec.c > > > create mode 100644 include/linux/platform_data/huawei-gaokun-ec.h > > > > > > diff --git a/MAINTAINERS b/MAINTAINERS > > > index 33b9cd11a..27ff24e7d 100644 > > > --- a/MAINTAINERS > > > +++ b/MAINTAINERS > > > @@ -10692,6 +10692,13 @@ S: Maintained > > > F: Documentation/networking/device_drivers/ethernet/huawei/hinic.rst > > > F: drivers/net/ethernet/huawei/hinic/ > > > > > > +HUAWEI MATEBOOK E GO EMBEDDED CONTROLLER DRIVER > > > +M: Pengyu Luo <mitltlatltl@xxxxxxxxx> > > > +S: Maintained > > > +F: Documentation/devicetree/bindings/platform/huawei,gaokun-ec.yaml > > > +F: drivers/platform/arm64/huawei-gaokun-ec.c > > > +F: include/linux/platform_data/huawei-gaokun-ec.h > > > + > > > HUGETLB SUBSYSTEM > > > M: Muchun Song <muchun.song@xxxxxxxxx> > > > L: linux-mm@xxxxxxxxx > > > diff --git a/drivers/platform/arm64/Kconfig b/drivers/platform/arm64/Kconfig > > > index f88395ea3..6ceee3c16 100644 > > > --- a/drivers/platform/arm64/Kconfig > > > +++ b/drivers/platform/arm64/Kconfig > > > @@ -33,6 +33,27 @@ config EC_ACER_ASPIRE1 > > > laptop where this information is not properly exposed via the > > > standard ACPI devices. > > > > > > +config EC_HUAWEI_GAOKUN > > > + tristate "Huawei Matebook E Go Embedded Controller driver" > > > + depends on ARCH_QCOM || COMPILE_TEST > > > + depends on I2C > > > + depends on INPUT > > > + select AUXILIARY_BUS > > > + > > > + help > > > + Say Y here to enable the EC driver for the Huawei Matebook E Go > > > + which is a sc8280xp-based 2-in-1 tablet. The driver handles battery > > > + (information, charge control) and USB Type-C DP HPD events as well > > > + as some misc functions like the lid sensor and temperature sensors, > > > + etc. > > > + > > > + This driver provides battery and AC status support for the mentioned > > > + laptop where this information is not properly exposed via the > > > + standard ACPI devices. > > > + > > > + Say M or Y here to include this support. > > > + > > > + > > > config EC_LENOVO_YOGA_C630 > > > tristate "Lenovo Yoga C630 Embedded Controller driver" > > > depends on ARCH_QCOM || COMPILE_TEST > > > diff --git a/drivers/platform/arm64/Makefile b/drivers/platform/arm64/Makefile > > > index b2ae9114f..46a99eba3 100644 > > > --- a/drivers/platform/arm64/Makefile > > > +++ b/drivers/platform/arm64/Makefile > > > @@ -6,4 +6,5 @@ > > > # > > > > > > obj-$(CONFIG_EC_ACER_ASPIRE1) += acer-aspire1-ec.o > > > +obj-$(CONFIG_EC_HUAWEI_GAOKUN) += huawei-gaokun-ec.o > > > obj-$(CONFIG_EC_LENOVO_YOGA_C630) += lenovo-yoga-c630.o > > > diff --git a/drivers/platform/arm64/huawei-gaokun-ec.c b/drivers/platform/arm64/huawei-gaokun-ec.c > > > new file mode 100644 > > > index 000000000..5b09b7d7c > > > --- /dev/null > > > +++ b/drivers/platform/arm64/huawei-gaokun-ec.c > > > @@ -0,0 +1,787 @@ > > > +// SPDX-License-Identifier: GPL-2.0-only > > > +/* > > > + * huawei-gaokun-ec - An EC driver for HUAWEI Matebook E Go > > > + * > > > + * Copyright (C) 2024-2025 Pengyu Luo <mitltlatltl@xxxxxxxxx> > > > + */ > > > + > > > +#include <linux/auxiliary_bus.h> > > > +#include <linux/cleanup.h> > > > +#include <linux/delay.h> > > > +#include <linux/device.h> > > > +#include <linux/hwmon.h> > > > +#include <linux/hwmon-sysfs.h> > > > +#include <linux/i2c.h> > > > +#include <linux/input.h> > > > +#include <linux/notifier.h> > > > +#include <linux/module.h> > > > +#include <linux/mutex.h> > > > +#include <linux/platform_data/huawei-gaokun-ec.h> > > > + > > > +#define EC_EVENT 0x06 > > > + > > > +/* Also can be found in ACPI specification 12.3 */ > > > +#define EC_READ 0x80 > > > +#define EC_WRITE 0x81 > > > +#define EC_BURST 0x82 > > > +#define EC_QUERY 0x84 > > > + > > > +#define EC_FN_LOCK_ON 0x5A > > > +#define EC_FN_LOCK_OFF 0x55 > > > + > > > +#define EC_EVENT_LID 0x81 > > > + > > > +#define EC_LID_STATE 0x80 > > > +#define EC_LID_OPEN BIT(1) > > > + > > > +#define UCSI_REG_SIZE 7 > > > + > > > +/* > > > + * For tx, command sequences are arranged as > > > + * {master_cmd, slave_cmd, data_len, data_seq} > > > + */ > > > +#define REQ_HDR_SIZE 3 > > > +#define INPUT_SIZE_OFFSET 2 > > > +#define REQ_LEN(req) (REQ_HDR_SIZE + req[INPUT_SIZE_OFFSET]) > > > + > > > +/* > > > + * For rx, data sequences are arranged as > > > + * {status, data_len(unreliable), data_seq} > > > + */ > > > +#define RESP_HDR_SIZE 2 > > > + > > > +#define MKREQ(REG0, REG1, SIZE, ...) \ > > > +{ \ > > > + REG0, REG1, SIZE, \ > > > + /* ## will remove comma when SIZE is 0 */ \ > > > + ## __VA_ARGS__, \ > > > + /* make sure len(pkt[3:]) >= SIZE */ \ > > > + [3 + SIZE] = 0, \ > > > +} > > > + > > > +#define MKRESP(SIZE) \ > > > +{ \ > > > + [RESP_HDR_SIZE + SIZE - 1] = 0, \ > > > +} > > > + > > > +/* Possible size 1, 4, 20, 24. Most of the time, the size is 1. */ > > > +static inline void refill_req(u8 *dest, const u8 *src, size_t size) > > > +{ > > > + memcpy(dest + REQ_HDR_SIZE, src, size); > > > +} > > > + > > > +static inline void refill_req_byte(u8 *dest, const u8 *src) > > > +{ > > > + dest[REQ_HDR_SIZE] = *src; > > > +} > > > + > > > +/* Possible size 1, 2, 4, 7, 20. Most of the time, the size is 1. */ > > > +static inline void extr_resp(u8 *dest, const u8 *src, size_t size) > > > +{ > > > + memcpy(dest, src + RESP_HDR_SIZE, size); > > > +} > > > + > > > +static inline void extr_resp_byte(u8 *dest, const u8 *src) > > > +{ > > > + *dest = src[RESP_HDR_SIZE]; > > > +} > > > + > > > +static inline void *extr_resp_shallow(const u8 *src) > > > +{ > > > + return src + RESP_HDR_SIZE; > > > +} > > > + > > > +struct gaokun_ec { > > > + struct i2c_client *client; > > > + struct mutex lock; /* EC transaction lock */ > > > + struct blocking_notifier_head notifier_list; > > > + struct device *hwmon_dev; > > > + struct input_dev *idev; > > > + bool suspended; > > > +}; > > > + > > > +static int gaokun_ec_request(struct gaokun_ec *ec, const u8 *req, > > > + size_t resp_len, u8 *resp) > > > +{ > > > + struct i2c_client *client = ec->client; > > > + struct i2c_msg msgs[2] = { > > > + { > > > + .addr = client->addr, > > > + .flags = client->flags, > > > + .len = REQ_LEN(req), > > > + .buf = req, > > > + }, { > > > + .addr = client->addr, > > > + .flags = client->flags | I2C_M_RD, > > > + .len = resp_len, > > > + .buf = resp, > > > + }, > > > + }; > > > + > > > + guard(mutex)(&ec->lock); > > > + i2c_transfer(client->adapter, msgs, ARRAY_SIZE(msgs)); > > > > You should trap the result code of i2c_transfer() and push it up the > > call stack. > > > > This EC uses SMBus Protocol, I guess. Qualcomm I2C driver doesn't support > this though. The response structure define by SMBus I mentioned them above > (Please also check ACPI specification 13.2.5) You can use i2c_smbus_write_byte_data and others with plain i2c controllers, They all are mentioned and described in Documentation/driver-api/i2c.rst I've used those APIs(with i2c controller in x1e80100) in my EC driver, and they worked fine > > +/* > + * For rx, data sequences are arranged as > + * {status, data_len(unreliable), data_seq} > + */ > > So the first byte is status code. > > > > + usleep_range(2000, 2500); /* have a break, ACPI did this */ > > > + > > > + return *resp ? -EIO : 0; > > > > If the value @ *resp is non-zero return -EIO ? > > > > Why ? > > > > Mentioned above. > > > > +} > > > + > > > +/* -------------------------------------------------------------------------- */ > > > +/* Common API */ > > > + > > > +/** > > > + * gaokun_ec_read - Read from EC > > > + * @ec: The gaokun_ec structure > > > + * @req: The sequence to request > > > + * @resp_len: The size to read > > > + * @resp: The buffer to store response sequence > > > + * > > > + * This function is used to read data after writing a magic sequence to EC. > > > + * All EC operations depend on this function. > > > + * > > > + * Huawei uses magic sequences everywhere to complete various functions, all > > > + * these sequences are passed to ECCD(a ACPI method which is quiet similar > > > + * to gaokun_ec_request), there is no good abstraction to generalize these > > > + * sequences, so just wrap it for now. Almost all magic sequences are kept > > > + * in this file. > > > + * > > > + * Return: 0 on success or negative error code. > > > + */ > > > +int gaokun_ec_read(struct gaokun_ec *ec, const u8 *req, > > > + size_t resp_len, u8 *resp) > > > +{ > > > + return gaokun_ec_request(ec, req, resp_len, resp); > > > +} > > > +EXPORT_SYMBOL_GPL(gaokun_ec_read); > > > + > > > +/** > > > + * gaokun_ec_write - Write to EC > > > + * @ec: The gaokun_ec structure > > > + * @req: The sequence to request > > > + * > > > + * This function has no big difference from gaokun_ec_read. When caller care > > > + * only write status and no actual data are returned, then use it. > > > + * > > > + * Return: 0 on success or negative error code. > > > + */ > > > +int gaokun_ec_write(struct gaokun_ec *ec, const u8 *req) > > > +{ > > > + u8 ec_resp[] = MKRESP(0); > > > + > > > + return gaokun_ec_request(ec, req, sizeof(ec_resp), ec_resp); > > > +} > > > +EXPORT_SYMBOL_GPL(gaokun_ec_write); > > > + > > > +int gaokun_ec_read_byte(struct gaokun_ec *ec, const u8 *req, u8 *byte) > > > +{ > > > + int ret; > > > + u8 ec_resp[] = MKRESP(sizeof(*byte)); > > > + > > > + ret = gaokun_ec_read(ec, req, sizeof(ec_resp), ec_resp); > > > + extr_resp_byte(byte, ec_resp); > > > + > > > + return ret; > > > +} > > > +EXPORT_SYMBOL_GPL(gaokun_ec_read_byte); > > > + > > > +/** > > > + * gaokun_ec_register_notify - Register a notifier callback for EC events. > > > + * @ec: The gaokun_ec structure > > > + * @nb: Notifier block pointer to register > > > + * > > > + * Return: 0 on success or negative error code. > > > + */ > > > +int gaokun_ec_register_notify(struct gaokun_ec *ec, struct notifier_block *nb) > > > +{ > > > + return blocking_notifier_chain_register(&ec->notifier_list, nb); > > > +} > > > +EXPORT_SYMBOL_GPL(gaokun_ec_register_notify); > > > + > > > +/** > > > + * gaokun_ec_unregister_notify - Unregister notifier callback for EC events. > > > + * @ec: The gaokun_ec structure > > > + * @nb: Notifier block pointer to unregister > > > + * > > > + * Unregister a notifier callback that was previously registered with > > > + * gaokun_ec_register_notify(). > > > + */ > > > +void gaokun_ec_unregister_notify(struct gaokun_ec *ec, struct notifier_block *nb) > > > +{ > > > + blocking_notifier_chain_unregister(&ec->notifier_list, nb); > > > +} > > > +EXPORT_SYMBOL_GPL(gaokun_ec_unregister_notify); > > > + > > > +/* -------------------------------------------------------------------------- */ > > > +/* API for PSY */ > > > + > > > +/** > > > + * gaokun_ec_psy_multi_read - Read contiguous registers > > > + * @ec: The gaokun_ec structure > > > + * @reg: The start register > > > + * @resp_len: The number of registers to be read > > > + * @resp: The buffer to store response sequence > > > + * > > > + * Return: 0 on success or negative error code. > > > + */ > > > +int gaokun_ec_psy_multi_read(struct gaokun_ec *ec, u8 reg, > > > + size_t resp_len, u8 *resp) > > > +{ > > > + u8 ec_req[] = MKREQ(0x02, EC_READ, 1, 0); > > > + u8 ec_resp[] = MKRESP(1); > > > + int i, ret; > > > + > > > + for (i = 0; i < resp_len; ++i, reg++) { > > > + refill_req_byte(ec_req, ®); > > > + ret = gaokun_ec_read(ec, ec_req, sizeof(ec_resp), ec_resp); > > > + if (ret) > > > + return ret; > > > + extr_resp_byte(&resp[i], ec_resp); > > > + } > > > + > > > + return 0; > > > +} > > > +EXPORT_SYMBOL_GPL(gaokun_ec_psy_multi_read); > > > + > > > +/* Smart charge */ > > > + > > > +/** > > > + * gaokun_ec_psy_get_smart_charge - Get smart charge data from EC > > > + * @ec: The gaokun_ec structure > > > + * @resp: The buffer to store response sequence (mode, delay, start, end) > > > + * > > > + * Return: 0 on success or negative error code. > > > + */ > > > +int gaokun_ec_psy_get_smart_charge(struct gaokun_ec *ec, > > > + u8 resp[GAOKUN_SMART_CHARGE_DATA_SIZE]) > > > +{ > > > + /* GBCM */ > > > + u8 ec_req[] = MKREQ(0x02, 0xE4, 0); > > > + u8 ec_resp[] = MKRESP(GAOKUN_SMART_CHARGE_DATA_SIZE); > > > + int ret; > > > + > > > + ret = gaokun_ec_read(ec, ec_req, sizeof(ec_resp), ec_resp); > > > + if (ret) > > > + return ret; > > > + > > > + extr_resp(resp, ec_resp, GAOKUN_SMART_CHARGE_DATA_SIZE); > > > + > > > + return 0; > > > +} > > > +EXPORT_SYMBOL_GPL(gaokun_ec_psy_get_smart_charge); > > > + > > > +static inline bool are_thresholds_valid(u8 start, u8 end) > > > +{ > > > + return end != 0 && start <= end && end <= 100; > > > > Why 100 ? Still feels like an arbitrary number. > > > > Could you add a comment to explain where 100 comes from ? > > > > You may don't get it. It is just a battery percentage, greater than 100 is > invalid. > > start: The battery percentage at which charging starts (0-100). > stop: The battery percentage at which charging stops (1-100). > > When the laptop is connected to a power adapter, it starts > charging if the battery level is below the 'start' value. It > continues charging until the battery reaches the 'stop' level. > If the battery is already above the 'stop' level, charging is > paused. > > When the power adapter is always connected, charging will > begin if the battery level falls below 'start', and charging > will stop once the battery reaches 'stop'. > > > > +} > > > + > > > +/** > > > + * gaokun_ec_psy_set_smart_charge - Set smart charge data > > > + * @ec: The gaokun_ec structure > > > + * @req: The sequence to request (mode, delay, start, end) > > > + * > > > + * Return: 0 on success or negative error code. > > > + */ > > > +int gaokun_ec_psy_set_smart_charge(struct gaokun_ec *ec, > > > + const u8 req[GAOKUN_SMART_CHARGE_DATA_SIZE]) > > > +{ > > > + /* SBCM */ > > > + u8 ec_req[] = MKREQ(0x02, 0xE3, GAOKUN_SMART_CHARGE_DATA_SIZE); > > > + > > > + if (!are_thresholds_valid(req[2], req[3])) > > > + return -EINVAL; > > > + > > > + refill_req(ec_req, req, GAOKUN_SMART_CHARGE_DATA_SIZE); > > > + > > > + return gaokun_ec_write(ec, ec_req); > > > +} > > > +EXPORT_SYMBOL_GPL(gaokun_ec_psy_set_smart_charge); > > > + > > > +/* Smart charge enable */ > > > + > > > +/** > > > + * gaokun_ec_psy_get_smart_charge_enable - Get smart charge state > > > + * @ec: The gaokun_ec structure > > > + * @on: The state > > > + * > > > + * Return: 0 on success or negative error code. > > > + */ > > > +int gaokun_ec_psy_get_smart_charge_enable(struct gaokun_ec *ec, bool *on) > > > +{ > > > + /* GBAC */ > > > + u8 ec_req[] = MKREQ(0x02, 0xE6, 0); > > > > 0xE6 == SMART_CHARGE_ENABLE right ? > > > > In which case instead of passing magic numbers inline, it would be nice > > to declare an enum or set of defines that enumerates the command set and > > then pass those values to MKREQ instead of the hex. > > > > Does the first parameter indicate write ? 0x02. 0x03 seems to indicate > > read too ? > > > > Sadly, I am not sure. only 0x06 is used when getting event id, so I named > it. Read or write use both 0x02, 0x03. > > > If so again please name the hex values as defines/enums and then pass > > the descriptive names. > > > > Function names indicate the usage of registers, these registers are used > once only, never reused. If you insist, I will add defines for them. > > > > Looking much more readable now though. > > > > --- > > bod > > > Best wishes, > Pengyu >