/**************************************************************************** * drivers/power/supply/regulator_rpmsg.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include <nuttx/config.h> #include <errno.h> #include <semaphore.h> #include <stdlib.h> #include <string.h> #include <sys/param.h> #include <nuttx/kmalloc.h> #include <nuttx/list.h> #include <nuttx/power/consumer.h> #include <nuttx/rptun/openamp.h> /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define REGULATOR_RPMSG_EPT_NAME "rpmsg-regulator" #define REGULATOR_RPMSG_ENABLE 0 #define REGULATOR_RPMSG_DISABLE 1 #define REGULATOR_RPMSG_GET_VOLTAGE 2 #define REGULATOR_RPMSG_SET_VOLTAGE 3 #define REGULATOR_RPMSG_IS_ENABLED 4 /**************************************************************************** * Private Types ****************************************************************************/ begin_packed_struct struct regulator_rpmsg_header_s { uint32_t command : 31; uint32_t response : 1; int32_t result; uint64_t cookie; } end_packed_struct; begin_packed_struct struct regulator_rpmsg_enable_s { struct regulator_rpmsg_header_s header; char name[0]; } end_packed_struct; #define regulator_rpmsg_disable_s regulator_rpmsg_enable_s #define regulator_rpmsg_isenabled_s regulator_rpmsg_enable_s #define regulator_rpmsg_getvol_s regulator_rpmsg_enable_s begin_packed_struct struct regulator_rpmsg_setvol_s { struct regulator_rpmsg_header_s header; int32_t min_uv; int32_t max_uv; char name[0]; } end_packed_struct; struct regulator_rpmsg_cookie_s { int32_t result; sem_t sem; }; struct regulator_rpmsg_client_s { struct rpmsg_endpoint ept; struct list_node node; sem_t sem; char cpuname[0]; }; struct regulator_rpmsg_server_s { struct rpmsg_endpoint ept; mutex_t lock; struct list_node regulator_list; }; struct regulator_rpmsg_s { FAR struct regulator_s *regulator; struct list_node node; bool enable; }; /**************************************************************************** * Private Function Prototypes ****************************************************************************/ static int regulator_rpmsg_ept_cb(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv); static int regulator_rpmsg_enable_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv); static int regulator_rpmsg_disable_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv); static int regulator_rpmsg_getvol_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv); static int regulator_rpmsg_setvol_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv); static int regulator_rpmsg_isenabled_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv); static void regulator_rpmsg_client_created(struct rpmsg_device *rdev, FAR void *priv); static void regulator_rpmsg_client_destroy(struct rpmsg_device *rdev, FAR void *priv); static void regulator_rpmsg_server_unbind(FAR struct rpmsg_endpoint *ept); static bool regulator_rpmsg_server_match(FAR struct rpmsg_device *rdev, FAR void *priv, FAR const char *name, uint32_t dest); static void regulator_rpmsg_server_bind(FAR struct rpmsg_device *rdev, FAR void *priv, FAR const char *name, uint32_t dest); static int regulator_rpmsg_set_voltage(FAR struct regulator_dev_s *rdev, int min_uv, int max_uv, FAR unsigned *selector); static int regulator_rpmsg_get_voltage(FAR struct regulator_dev_s *rdev); static int regulator_rpmsg_enable(FAR struct regulator_dev_s *rdev); static int regulator_rpmsg_disable(FAR struct regulator_dev_s *rdev); static int regulator_rpmsg_is_enabled(FAR struct regulator_dev_s *rdev); /**************************************************************************** * Private Data ****************************************************************************/ static mutex_t g_regulator_rpmsg_lock = NXMUTEX_INITIALIZER; static struct list_node g_regulator_rpmsg_client = LIST_INITIAL_VALUE(g_regulator_rpmsg_client); static const rpmsg_ept_cb g_regulator_rpmsg_handler[] = { [REGULATOR_RPMSG_ENABLE] = regulator_rpmsg_enable_handler, [REGULATOR_RPMSG_DISABLE] = regulator_rpmsg_disable_handler, [REGULATOR_RPMSG_GET_VOLTAGE] = regulator_rpmsg_getvol_handler, [REGULATOR_RPMSG_SET_VOLTAGE] = regulator_rpmsg_setvol_handler, [REGULATOR_RPMSG_IS_ENABLED] = regulator_rpmsg_isenabled_handler, }; static const struct regulator_ops_s g_regulator_rpmsg_ops = { .set_voltage = regulator_rpmsg_set_voltage, .get_voltage = regulator_rpmsg_get_voltage, .enable = regulator_rpmsg_enable, .disable = regulator_rpmsg_disable, .is_enabled = regulator_rpmsg_is_enabled, }; /**************************************************************************** * Private Functions ****************************************************************************/ static FAR struct regulator_rpmsg_client_s * regulator_rpmsg_get_client(FAR const char *name) { FAR struct regulator_rpmsg_client_s *client; FAR const char *slash = strchr(name, '/'); if (slash == NULL) { return NULL; } nxmutex_lock(&g_regulator_rpmsg_lock); list_for_every_entry(&g_regulator_rpmsg_client, client, struct regulator_rpmsg_client_s, node) { if (strncmp(client->cpuname, name, slash - name) == 0) { goto out; /* Find the target, exit */ } } client = kmm_zalloc(sizeof(*client) + slash - name + 1); if (client == NULL) { goto out; } memcpy(client->cpuname, name, slash - name); list_add_head(&g_regulator_rpmsg_client, &client->node); nxmutex_unlock(&g_regulator_rpmsg_lock); rpmsg_register_callback(client, regulator_rpmsg_client_created, regulator_rpmsg_client_destroy, NULL, NULL); return client; out: nxmutex_unlock(&g_regulator_rpmsg_lock); return client; } static FAR struct rpmsg_endpoint * regulator_rpmsg_get_ept(FAR const char **name) { FAR struct regulator_rpmsg_client_s *client; int ret = 0; client = regulator_rpmsg_get_client(*name); if (client == NULL) { return NULL; } if (!is_rpmsg_ept_ready(&client->ept)) { ret = nxsem_wait_uninterruptible(&client->sem); if (ret < 0) { return NULL; } nxsem_post(&client->sem); } *name += strlen(client->cpuname) + 1; return &client->ept; } static FAR struct regulator_rpmsg_s * regulator_rpmsg_get_reg(FAR struct rpmsg_endpoint *ept, FAR const char *name) { FAR struct regulator_rpmsg_server_s *server = ept->priv; FAR struct list_node *regulator_list = &server->regulator_list; FAR struct regulator_rpmsg_s *reg; nxmutex_lock(&server->lock); list_for_every_entry(regulator_list, reg, struct regulator_rpmsg_s, node) { if (strcmp(reg->regulator->rdev->desc->name, name) == 0) { nxmutex_unlock(&server->lock); return reg; } } reg = kmm_zalloc(sizeof(*reg)); if (reg == NULL) { nxmutex_unlock(&server->lock); return NULL; } reg->regulator = regulator_get(name); if (reg->regulator == NULL) { kmm_free(reg); nxmutex_unlock(&server->lock); return NULL; } list_add_head(regulator_list, ®->node); nxmutex_unlock(&server->lock); return reg; } static void regulator_rpmsg_client_created(struct rpmsg_device *rdev, FAR void *priv) { FAR struct regulator_rpmsg_client_s *client = priv; if (client == NULL) { return; } if (strcmp(client->cpuname, rpmsg_get_cpuname(rdev)) == 0) { client->ept.priv = client; rpmsg_create_ept(&client->ept, rdev, REGULATOR_RPMSG_EPT_NAME, RPMSG_ADDR_ANY, RPMSG_ADDR_ANY, regulator_rpmsg_ept_cb, NULL); nxsem_post(&client->sem); } } static void regulator_rpmsg_client_destroy(struct rpmsg_device *rdev, FAR void *priv) { FAR struct regulator_rpmsg_client_s *client = priv; if (client == NULL) { return; } if (strcmp(client->cpuname, rpmsg_get_cpuname(rdev)) == 0) { nxsem_wait(&client->sem); rpmsg_destroy_ept(&client->ept); } } static void regulator_rpmsg_server_unbind(FAR struct rpmsg_endpoint *ept) { FAR struct regulator_rpmsg_server_s *server = ept->priv; FAR struct regulator_rpmsg_s *reg; FAR struct regulator_rpmsg_s *tmp; list_for_every_entry_safe(&server->regulator_list, reg, tmp, struct regulator_rpmsg_s, node) { if (reg->enable) { regulator_disable(reg->regulator); } regulator_put(reg->regulator); list_delete(®->node); kmm_free(reg); } nxmutex_destroy(&server->lock); rpmsg_destroy_ept(ept); kmm_free(server); } static bool regulator_rpmsg_server_match(FAR struct rpmsg_device *rdev, FAR void *priv, FAR const char *name, uint32_t dest) { return strcmp(name, REGULATOR_RPMSG_EPT_NAME) == 0; } static void regulator_rpmsg_server_bind(FAR struct rpmsg_device *rdev, FAR void *priv, FAR const char *name, uint32_t dest) { FAR struct regulator_rpmsg_server_s *server; server = kmm_zalloc(sizeof(*server)); if (server == NULL) { return; } server->ept.priv = server; nxmutex_init(&server->lock); list_initialize(&server->regulator_list); rpmsg_create_ept(&server->ept, rdev, name, RPMSG_ADDR_ANY, RPMSG_ADDR_ANY, regulator_rpmsg_ept_cb, regulator_rpmsg_server_unbind); } static int regulator_rpmsg_ept_cb(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv) { FAR struct regulator_rpmsg_header_s *header = data; FAR struct regulator_rpmsg_cookie_s *cookie = (FAR struct regulator_rpmsg_cookie_s *)(uintptr_t)header->cookie; uint32_t cmd = header->command; int ret = -EINVAL; if (cookie && header->response) { cookie->result = header->result; nxsem_post(&cookie->sem); ret = 0; } else if (cmd < nitems(g_regulator_rpmsg_handler) && g_regulator_rpmsg_handler[cmd]) { header->response = 1; ret = g_regulator_rpmsg_handler[cmd](ept, data, len, src, priv); } return ret; } static int regulator_rpmsg_enable_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv) { FAR struct regulator_rpmsg_enable_s *msg = data; FAR struct regulator_rpmsg_s *reg = regulator_rpmsg_get_reg(ept, msg->name); int ret = -ENOENT; if (reg && !reg->enable) { ret = regulator_enable(reg->regulator); if (ret >= 0) { reg->enable = true; } } else if (reg && reg->enable) { ret = 0; } msg->header.result = ret; return rpmsg_send(ept, data, len); } static int regulator_rpmsg_disable_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv) { FAR struct regulator_rpmsg_disable_s *msg = data; FAR struct regulator_rpmsg_s *reg = regulator_rpmsg_get_reg(ept, msg->name); int ret = -ENOENT; if (reg && reg->enable) { ret = regulator_disable(reg->regulator); if (ret >= 0) { reg->enable = false; } } else if (reg && !reg->enable) { ret = 0; } msg->header.result = ret; return rpmsg_send(ept, data, len); } static int regulator_rpmsg_getvol_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv) { FAR struct regulator_rpmsg_getvol_s *msg = data; FAR struct regulator_rpmsg_s *reg = regulator_rpmsg_get_reg(ept, msg->name); int ret = -ENOENT; if (reg) { ret = regulator_get_voltage(reg->regulator); } msg->header.result = ret; return rpmsg_send(ept, data, len); } static int regulator_rpmsg_setvol_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv) { FAR struct regulator_rpmsg_setvol_s *msg = data; FAR struct regulator_rpmsg_s *reg = regulator_rpmsg_get_reg(ept, msg->name); int ret = -ENOENT; if (reg) { ret = regulator_set_voltage(reg->regulator, msg->min_uv, msg->max_uv); } msg->header.result = ret; return rpmsg_send(ept, data, len); } static int regulator_rpmsg_isenabled_handler(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv) { FAR struct regulator_rpmsg_isenabled_s *msg = data; FAR struct regulator_rpmsg_s *reg = regulator_rpmsg_get_reg(ept, msg->name); int ret = -ENOENT; if (reg) { ret = reg->enable; } msg->header.result = ret; return rpmsg_send(ept, data, len); } static int regulator_rpmsg_sendrecv(FAR struct rpmsg_endpoint *ept, uint32_t command, FAR struct regulator_rpmsg_header_s *msg, int len) { struct regulator_rpmsg_cookie_s cookie = { 0 }; int ret; nxsem_init(&cookie.sem, 0, 0); msg->command = command; msg->response = 0; msg->result = -ENXIO; msg->cookie = (uintptr_t)&cookie; ret = rpmsg_send_nocopy(ept, msg, len); if (ret >= 0) { ret = nxsem_wait_uninterruptible(&cookie.sem); if (ret >= 0) { ret = cookie.result; } } nxsem_destroy(&cookie.sem); return ret; } static int regulator_rpmsg_enable(FAR struct regulator_dev_s *rdev) { FAR struct rpmsg_endpoint *ept; FAR struct regulator_rpmsg_enable_s *msg; FAR const char *name = rdev->desc->name; uint32_t len; ept = regulator_rpmsg_get_ept(&name); if (ept == NULL) { return -ENODEV; } len = sizeof(*msg) + strlen(name) + 1; msg = rpmsg_get_tx_payload_buffer(ept, &len, true); if (msg == NULL) { return -ENOMEM; } strlcpy(msg->name, name, len - sizeof(*msg)); return regulator_rpmsg_sendrecv(ept, REGULATOR_RPMSG_ENABLE, (FAR struct regulator_rpmsg_header_s *)msg, len); } static int regulator_rpmsg_disable(FAR struct regulator_dev_s *rdev) { FAR struct rpmsg_endpoint *ept; FAR struct regulator_rpmsg_disable_s *msg; FAR const char *name = rdev->desc->name; uint32_t len; ept = regulator_rpmsg_get_ept(&name); if (ept == NULL) { return -ENODEV; } len = sizeof(*msg) + strlen(name) + 1; msg = rpmsg_get_tx_payload_buffer(ept, &len, true); if (msg == NULL) { return -ENOMEM; } strlcpy(msg->name, name, len - sizeof(*msg)); return regulator_rpmsg_sendrecv(ept, REGULATOR_RPMSG_DISABLE, (FAR struct regulator_rpmsg_header_s *)msg, len); } static int regulator_rpmsg_set_voltage(FAR struct regulator_dev_s *rdev, int min_uv, int max_uv, FAR unsigned *selector) { FAR struct rpmsg_endpoint *ept; FAR struct regulator_rpmsg_setvol_s *msg; FAR const char *name = rdev->desc->name; uint32_t len; ept = regulator_rpmsg_get_ept(&name); if (ept == NULL) { return -ENODEV; } len = sizeof(*msg) + strlen(name) + 1; msg = rpmsg_get_tx_payload_buffer(ept, &len, true); if (msg == NULL) { return -ENOMEM; } strlcpy(msg->name, name, len - sizeof(*msg)); msg->min_uv = min_uv; msg->max_uv = max_uv; return regulator_rpmsg_sendrecv(ept, REGULATOR_RPMSG_SET_VOLTAGE, (FAR struct regulator_rpmsg_header_s *)msg, len); } static int regulator_rpmsg_get_voltage(FAR struct regulator_dev_s *rdev) { FAR struct rpmsg_endpoint *ept; FAR struct regulator_rpmsg_getvol_s *msg; FAR const char *name = rdev->desc->name; uint32_t len; ept = regulator_rpmsg_get_ept(&name); if (ept == NULL) { return -ENODEV; } len = sizeof(*msg) + strlen(name) + 1; msg = rpmsg_get_tx_payload_buffer(ept, &len, true); if (msg == NULL) { return -ENOMEM; } strlcpy(msg->name, name, len - sizeof(*msg)); return regulator_rpmsg_sendrecv(ept, REGULATOR_RPMSG_GET_VOLTAGE, (FAR struct regulator_rpmsg_header_s *)msg, len); } static int regulator_rpmsg_is_enabled(FAR struct regulator_dev_s *rdev) { FAR struct rpmsg_endpoint *ept; FAR struct regulator_rpmsg_isenabled_s *msg; FAR const char *name = rdev->desc->name; uint32_t len; ept = regulator_rpmsg_get_ept(&name); if (ept == NULL) { return -ENODEV; } len = sizeof(*msg) + strlen(name) + 1; msg = rpmsg_get_tx_payload_buffer(ept, &len, true); if (msg == NULL) { return -ENOMEM; } strlcpy(msg->name, name, len - sizeof(*msg)); return regulator_rpmsg_sendrecv(ept, REGULATOR_RPMSG_IS_ENABLED, (struct regulator_rpmsg_header_s *)msg, len); } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: regulator_rpmsg_get * * Description: * * Input Parameters: * * name - the name for register the rpmsg regulator dev * * Returned Value: * * Regulator dev pointer * ****************************************************************************/ FAR struct regulator_dev_s *regulator_rpmsg_get(FAR const char *name) { FAR struct regulator_desc_s *desc; FAR struct regulator_dev_s *dev; FAR char *pname; size_t len = strlen(name) + 1; desc = kmm_zalloc(sizeof(*desc) + len); if (desc == NULL) { return NULL; } pname = (FAR char *)(desc + 1); memcpy(pname, name, len); desc->name = pname; dev = regulator_register(desc, &g_regulator_rpmsg_ops, NULL); if (dev == NULL) { kmm_free(desc); } return dev; } /**************************************************************************** * Name: regulator_rpmsg_server_init * * Description: * * Establish rpmsg channel for the operations of the remote regulator * * Input Parameters: * * Returned Value: * Zero (OK) on success; a negated errno on failure * ****************************************************************************/ int regulator_rpmsg_server_init(void) { return rpmsg_register_callback(NULL, NULL, NULL, regulator_rpmsg_server_match, regulator_rpmsg_server_bind); }