// SPDX-License-Identifier: GPL-2.0
|
/*
|
* Greybus Lights protocol driver.
|
*
|
* Copyright 2015 Google Inc.
|
* Copyright 2015 Linaro Ltd.
|
*/
|
|
#include <linux/kernel.h>
|
#include <linux/leds.h>
|
#include <linux/led-class-flash.h>
|
#include <linux/module.h>
|
#include <linux/slab.h>
|
#include <linux/greybus.h>
|
#include <media/v4l2-flash-led-class.h>
|
|
#define NAMES_MAX 32
|
|
struct gb_channel {
|
u8 id;
|
u32 flags;
|
u32 color;
|
char *color_name;
|
u8 fade_in;
|
u8 fade_out;
|
u32 mode;
|
char *mode_name;
|
struct attribute **attrs;
|
struct attribute_group *attr_group;
|
const struct attribute_group **attr_groups;
|
struct led_classdev *led;
|
#if IS_REACHABLE(CONFIG_LEDS_CLASS_FLASH)
|
struct led_classdev_flash fled;
|
struct led_flash_setting intensity_uA;
|
struct led_flash_setting timeout_us;
|
#else
|
struct led_classdev cled;
|
#endif
|
struct gb_light *light;
|
bool is_registered;
|
bool releasing;
|
bool strobe_state;
|
bool active;
|
struct mutex lock;
|
};
|
|
struct gb_light {
|
u8 id;
|
char *name;
|
struct gb_lights *glights;
|
u32 flags;
|
u8 channels_count;
|
struct gb_channel *channels;
|
bool has_flash;
|
bool ready;
|
#if IS_REACHABLE(CONFIG_V4L2_FLASH_LED_CLASS)
|
struct v4l2_flash *v4l2_flash;
|
struct v4l2_flash *v4l2_flash_ind;
|
#endif
|
};
|
|
struct gb_lights {
|
struct gb_connection *connection;
|
u8 lights_count;
|
struct gb_light *lights;
|
struct mutex lights_lock;
|
};
|
|
static void gb_lights_channel_free(struct gb_channel *channel);
|
|
static struct gb_connection *get_conn_from_channel(struct gb_channel *channel)
|
{
|
return channel->light->glights->connection;
|
}
|
|
static struct gb_connection *get_conn_from_light(struct gb_light *light)
|
{
|
return light->glights->connection;
|
}
|
|
static bool is_channel_flash(struct gb_channel *channel)
|
{
|
return !!(channel->mode & (GB_CHANNEL_MODE_FLASH | GB_CHANNEL_MODE_TORCH
|
| GB_CHANNEL_MODE_INDICATOR));
|
}
|
|
#if IS_REACHABLE(CONFIG_LEDS_CLASS_FLASH)
|
static struct gb_channel *get_channel_from_cdev(struct led_classdev *cdev)
|
{
|
struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(cdev);
|
|
return container_of(fled_cdev, struct gb_channel, fled);
|
}
|
|
static struct led_classdev *get_channel_cdev(struct gb_channel *channel)
|
{
|
return &channel->fled.led_cdev;
|
}
|
|
static struct gb_channel *get_channel_from_mode(struct gb_light *light,
|
u32 mode)
|
{
|
struct gb_channel *channel = NULL;
|
int i;
|
|
for (i = 0; i < light->channels_count; i++) {
|
channel = &light->channels[i];
|
if (channel && channel->mode == mode)
|
break;
|
}
|
return channel;
|
}
|
|
static int __gb_lights_flash_intensity_set(struct gb_channel *channel,
|
u32 intensity)
|
{
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_bundle *bundle = connection->bundle;
|
struct gb_lights_set_flash_intensity_request req;
|
int ret;
|
|
if (channel->releasing)
|
return -ESHUTDOWN;
|
|
ret = gb_pm_runtime_get_sync(bundle);
|
if (ret < 0)
|
return ret;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
req.intensity_uA = cpu_to_le32(intensity);
|
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_SET_FLASH_INTENSITY,
|
&req, sizeof(req), NULL, 0);
|
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
return ret;
|
}
|
|
static int __gb_lights_flash_brightness_set(struct gb_channel *channel)
|
{
|
u32 intensity;
|
|
/* If the channel is flash we need to get the attached torch channel */
|
if (channel->mode & GB_CHANNEL_MODE_FLASH)
|
channel = get_channel_from_mode(channel->light,
|
GB_CHANNEL_MODE_TORCH);
|
|
/* For not flash we need to convert brightness to intensity */
|
intensity = channel->intensity_uA.min +
|
(channel->intensity_uA.step * channel->led->brightness);
|
|
return __gb_lights_flash_intensity_set(channel, intensity);
|
}
|
#else
|
static struct gb_channel *get_channel_from_cdev(struct led_classdev *cdev)
|
{
|
return container_of(cdev, struct gb_channel, cled);
|
}
|
|
static struct led_classdev *get_channel_cdev(struct gb_channel *channel)
|
{
|
return &channel->cled;
|
}
|
|
static int __gb_lights_flash_brightness_set(struct gb_channel *channel)
|
{
|
return 0;
|
}
|
#endif
|
|
static int gb_lights_color_set(struct gb_channel *channel, u32 color);
|
static int gb_lights_fade_set(struct gb_channel *channel);
|
|
static void led_lock(struct led_classdev *cdev)
|
{
|
mutex_lock(&cdev->led_access);
|
}
|
|
static void led_unlock(struct led_classdev *cdev)
|
{
|
mutex_unlock(&cdev->led_access);
|
}
|
|
#define gb_lights_fade_attr(__dir) \
|
static ssize_t fade_##__dir##_show(struct device *dev, \
|
struct device_attribute *attr, \
|
char *buf) \
|
{ \
|
struct led_classdev *cdev = dev_get_drvdata(dev); \
|
struct gb_channel *channel = get_channel_from_cdev(cdev); \
|
\
|
return sprintf(buf, "%u\n", channel->fade_##__dir); \
|
} \
|
\
|
static ssize_t fade_##__dir##_store(struct device *dev, \
|
struct device_attribute *attr, \
|
const char *buf, size_t size) \
|
{ \
|
struct led_classdev *cdev = dev_get_drvdata(dev); \
|
struct gb_channel *channel = get_channel_from_cdev(cdev); \
|
u8 fade; \
|
int ret; \
|
\
|
led_lock(cdev); \
|
if (led_sysfs_is_disabled(cdev)) { \
|
ret = -EBUSY; \
|
goto unlock; \
|
} \
|
\
|
ret = kstrtou8(buf, 0, &fade); \
|
if (ret < 0) { \
|
dev_err(dev, "could not parse fade value %d\n", ret); \
|
goto unlock; \
|
} \
|
if (channel->fade_##__dir == fade) \
|
goto unlock; \
|
channel->fade_##__dir = fade; \
|
\
|
ret = gb_lights_fade_set(channel); \
|
if (ret < 0) \
|
goto unlock; \
|
\
|
ret = size; \
|
unlock: \
|
led_unlock(cdev); \
|
return ret; \
|
} \
|
static DEVICE_ATTR_RW(fade_##__dir)
|
|
gb_lights_fade_attr(in);
|
gb_lights_fade_attr(out);
|
|
static ssize_t color_show(struct device *dev, struct device_attribute *attr,
|
char *buf)
|
{
|
struct led_classdev *cdev = dev_get_drvdata(dev);
|
struct gb_channel *channel = get_channel_from_cdev(cdev);
|
|
return sprintf(buf, "0x%08x\n", channel->color);
|
}
|
|
static ssize_t color_store(struct device *dev, struct device_attribute *attr,
|
const char *buf, size_t size)
|
{
|
struct led_classdev *cdev = dev_get_drvdata(dev);
|
struct gb_channel *channel = get_channel_from_cdev(cdev);
|
u32 color;
|
int ret;
|
|
led_lock(cdev);
|
if (led_sysfs_is_disabled(cdev)) {
|
ret = -EBUSY;
|
goto unlock;
|
}
|
ret = kstrtou32(buf, 0, &color);
|
if (ret < 0) {
|
dev_err(dev, "could not parse color value %d\n", ret);
|
goto unlock;
|
}
|
|
ret = gb_lights_color_set(channel, color);
|
if (ret < 0)
|
goto unlock;
|
|
channel->color = color;
|
ret = size;
|
unlock:
|
led_unlock(cdev);
|
return ret;
|
}
|
static DEVICE_ATTR_RW(color);
|
|
static int channel_attr_groups_set(struct gb_channel *channel,
|
struct led_classdev *cdev)
|
{
|
int attr = 0;
|
int size = 0;
|
|
if (channel->flags & GB_LIGHT_CHANNEL_MULTICOLOR)
|
size++;
|
if (channel->flags & GB_LIGHT_CHANNEL_FADER)
|
size += 2;
|
|
if (!size)
|
return 0;
|
|
/* Set attributes based in the channel flags */
|
channel->attrs = kcalloc(size + 1, sizeof(*channel->attrs), GFP_KERNEL);
|
if (!channel->attrs)
|
return -ENOMEM;
|
channel->attr_group = kcalloc(1, sizeof(*channel->attr_group),
|
GFP_KERNEL);
|
if (!channel->attr_group)
|
return -ENOMEM;
|
channel->attr_groups = kcalloc(2, sizeof(*channel->attr_groups),
|
GFP_KERNEL);
|
if (!channel->attr_groups)
|
return -ENOMEM;
|
|
if (channel->flags & GB_LIGHT_CHANNEL_MULTICOLOR)
|
channel->attrs[attr++] = &dev_attr_color.attr;
|
if (channel->flags & GB_LIGHT_CHANNEL_FADER) {
|
channel->attrs[attr++] = &dev_attr_fade_in.attr;
|
channel->attrs[attr++] = &dev_attr_fade_out.attr;
|
}
|
|
channel->attr_group->attrs = channel->attrs;
|
|
channel->attr_groups[0] = channel->attr_group;
|
|
cdev->groups = channel->attr_groups;
|
|
return 0;
|
}
|
|
static int gb_lights_fade_set(struct gb_channel *channel)
|
{
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_bundle *bundle = connection->bundle;
|
struct gb_lights_set_fade_request req;
|
int ret;
|
|
if (channel->releasing)
|
return -ESHUTDOWN;
|
|
ret = gb_pm_runtime_get_sync(bundle);
|
if (ret < 0)
|
return ret;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
req.fade_in = channel->fade_in;
|
req.fade_out = channel->fade_out;
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_SET_FADE,
|
&req, sizeof(req), NULL, 0);
|
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
return ret;
|
}
|
|
static int gb_lights_color_set(struct gb_channel *channel, u32 color)
|
{
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_bundle *bundle = connection->bundle;
|
struct gb_lights_set_color_request req;
|
int ret;
|
|
if (channel->releasing)
|
return -ESHUTDOWN;
|
|
ret = gb_pm_runtime_get_sync(bundle);
|
if (ret < 0)
|
return ret;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
req.color = cpu_to_le32(color);
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_SET_COLOR,
|
&req, sizeof(req), NULL, 0);
|
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
return ret;
|
}
|
|
static int __gb_lights_led_brightness_set(struct gb_channel *channel)
|
{
|
struct gb_lights_set_brightness_request req;
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_bundle *bundle = connection->bundle;
|
bool old_active;
|
int ret;
|
|
mutex_lock(&channel->lock);
|
ret = gb_pm_runtime_get_sync(bundle);
|
if (ret < 0)
|
goto out_unlock;
|
|
old_active = channel->active;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
req.brightness = (u8)channel->led->brightness;
|
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_SET_BRIGHTNESS,
|
&req, sizeof(req), NULL, 0);
|
if (ret < 0)
|
goto out_pm_put;
|
|
if (channel->led->brightness)
|
channel->active = true;
|
else
|
channel->active = false;
|
|
/* we need to keep module alive when turning to active state */
|
if (!old_active && channel->active)
|
goto out_unlock;
|
|
/*
|
* on the other hand if going to inactive we still hold a reference and
|
* need to put it, so we could go to suspend.
|
*/
|
if (old_active && !channel->active)
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
out_pm_put:
|
gb_pm_runtime_put_autosuspend(bundle);
|
out_unlock:
|
mutex_unlock(&channel->lock);
|
|
return ret;
|
}
|
|
static int __gb_lights_brightness_set(struct gb_channel *channel)
|
{
|
int ret;
|
|
if (channel->releasing)
|
return 0;
|
|
if (is_channel_flash(channel))
|
ret = __gb_lights_flash_brightness_set(channel);
|
else
|
ret = __gb_lights_led_brightness_set(channel);
|
|
return ret;
|
}
|
|
static int gb_brightness_set(struct led_classdev *cdev,
|
enum led_brightness value)
|
{
|
struct gb_channel *channel = get_channel_from_cdev(cdev);
|
|
channel->led->brightness = value;
|
|
return __gb_lights_brightness_set(channel);
|
}
|
|
static enum led_brightness gb_brightness_get(struct led_classdev *cdev)
|
|
{
|
struct gb_channel *channel = get_channel_from_cdev(cdev);
|
|
return channel->led->brightness;
|
}
|
|
static int gb_blink_set(struct led_classdev *cdev, unsigned long *delay_on,
|
unsigned long *delay_off)
|
{
|
struct gb_channel *channel = get_channel_from_cdev(cdev);
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_bundle *bundle = connection->bundle;
|
struct gb_lights_blink_request req;
|
bool old_active;
|
int ret;
|
|
if (channel->releasing)
|
return -ESHUTDOWN;
|
|
if (!delay_on || !delay_off)
|
return -EINVAL;
|
|
mutex_lock(&channel->lock);
|
ret = gb_pm_runtime_get_sync(bundle);
|
if (ret < 0)
|
goto out_unlock;
|
|
old_active = channel->active;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
req.time_on_ms = cpu_to_le16(*delay_on);
|
req.time_off_ms = cpu_to_le16(*delay_off);
|
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_SET_BLINK, &req,
|
sizeof(req), NULL, 0);
|
if (ret < 0)
|
goto out_pm_put;
|
|
if (*delay_on)
|
channel->active = true;
|
else
|
channel->active = false;
|
|
/* we need to keep module alive when turning to active state */
|
if (!old_active && channel->active)
|
goto out_unlock;
|
|
/*
|
* on the other hand if going to inactive we still hold a reference and
|
* need to put it, so we could go to suspend.
|
*/
|
if (old_active && !channel->active)
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
out_pm_put:
|
gb_pm_runtime_put_autosuspend(bundle);
|
out_unlock:
|
mutex_unlock(&channel->lock);
|
|
return ret;
|
}
|
|
static void gb_lights_led_operations_set(struct gb_channel *channel,
|
struct led_classdev *cdev)
|
{
|
cdev->brightness_get = gb_brightness_get;
|
cdev->brightness_set_blocking = gb_brightness_set;
|
|
if (channel->flags & GB_LIGHT_CHANNEL_BLINK)
|
cdev->blink_set = gb_blink_set;
|
}
|
|
#if IS_REACHABLE(CONFIG_V4L2_FLASH_LED_CLASS)
|
/* V4L2 specific helpers */
|
static const struct v4l2_flash_ops v4l2_flash_ops;
|
|
static void __gb_lights_channel_v4l2_config(struct led_flash_setting *channel_s,
|
struct led_flash_setting *v4l2_s)
|
{
|
v4l2_s->min = channel_s->min;
|
v4l2_s->max = channel_s->max;
|
v4l2_s->step = channel_s->step;
|
/* For v4l2 val is the default value */
|
v4l2_s->val = channel_s->max;
|
}
|
|
static int gb_lights_light_v4l2_register(struct gb_light *light)
|
{
|
struct gb_connection *connection = get_conn_from_light(light);
|
struct device *dev = &connection->bundle->dev;
|
struct v4l2_flash_config sd_cfg = { {0} }, sd_cfg_ind = { {0} };
|
struct led_classdev_flash *fled;
|
struct led_classdev *iled = NULL;
|
struct gb_channel *channel_torch, *channel_ind, *channel_flash;
|
|
channel_torch = get_channel_from_mode(light, GB_CHANNEL_MODE_TORCH);
|
if (channel_torch)
|
__gb_lights_channel_v4l2_config(&channel_torch->intensity_uA,
|
&sd_cfg.intensity);
|
|
channel_ind = get_channel_from_mode(light, GB_CHANNEL_MODE_INDICATOR);
|
if (channel_ind) {
|
__gb_lights_channel_v4l2_config(&channel_ind->intensity_uA,
|
&sd_cfg_ind.intensity);
|
iled = &channel_ind->fled.led_cdev;
|
}
|
|
channel_flash = get_channel_from_mode(light, GB_CHANNEL_MODE_FLASH);
|
WARN_ON(!channel_flash);
|
|
fled = &channel_flash->fled;
|
|
snprintf(sd_cfg.dev_name, sizeof(sd_cfg.dev_name), "%s", light->name);
|
snprintf(sd_cfg_ind.dev_name, sizeof(sd_cfg_ind.dev_name),
|
"%s indicator", light->name);
|
|
/* Set the possible values to faults, in our case all faults */
|
sd_cfg.flash_faults = LED_FAULT_OVER_VOLTAGE | LED_FAULT_TIMEOUT |
|
LED_FAULT_OVER_TEMPERATURE | LED_FAULT_SHORT_CIRCUIT |
|
LED_FAULT_OVER_CURRENT | LED_FAULT_INDICATOR |
|
LED_FAULT_UNDER_VOLTAGE | LED_FAULT_INPUT_VOLTAGE |
|
LED_FAULT_LED_OVER_TEMPERATURE;
|
|
light->v4l2_flash = v4l2_flash_init(dev, NULL, fled, &v4l2_flash_ops,
|
&sd_cfg);
|
if (IS_ERR(light->v4l2_flash))
|
return PTR_ERR(light->v4l2_flash);
|
|
if (channel_ind) {
|
light->v4l2_flash_ind =
|
v4l2_flash_indicator_init(dev, NULL, iled, &sd_cfg_ind);
|
if (IS_ERR(light->v4l2_flash_ind)) {
|
v4l2_flash_release(light->v4l2_flash);
|
return PTR_ERR(light->v4l2_flash_ind);
|
}
|
}
|
|
return 0;
|
}
|
|
static void gb_lights_light_v4l2_unregister(struct gb_light *light)
|
{
|
v4l2_flash_release(light->v4l2_flash_ind);
|
v4l2_flash_release(light->v4l2_flash);
|
}
|
#else
|
static int gb_lights_light_v4l2_register(struct gb_light *light)
|
{
|
struct gb_connection *connection = get_conn_from_light(light);
|
|
dev_err(&connection->bundle->dev, "no support for v4l2 subdevices\n");
|
return 0;
|
}
|
|
static void gb_lights_light_v4l2_unregister(struct gb_light *light)
|
{
|
}
|
#endif
|
|
#if IS_REACHABLE(CONFIG_LEDS_CLASS_FLASH)
|
/* Flash specific operations */
|
static int gb_lights_flash_intensity_set(struct led_classdev_flash *fcdev,
|
u32 brightness)
|
{
|
struct gb_channel *channel = container_of(fcdev, struct gb_channel,
|
fled);
|
int ret;
|
|
ret = __gb_lights_flash_intensity_set(channel, brightness);
|
if (ret < 0)
|
return ret;
|
|
fcdev->brightness.val = brightness;
|
|
return 0;
|
}
|
|
static int gb_lights_flash_intensity_get(struct led_classdev_flash *fcdev,
|
u32 *brightness)
|
{
|
*brightness = fcdev->brightness.val;
|
|
return 0;
|
}
|
|
static int gb_lights_flash_strobe_set(struct led_classdev_flash *fcdev,
|
bool state)
|
{
|
struct gb_channel *channel = container_of(fcdev, struct gb_channel,
|
fled);
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_bundle *bundle = connection->bundle;
|
struct gb_lights_set_flash_strobe_request req;
|
int ret;
|
|
if (channel->releasing)
|
return -ESHUTDOWN;
|
|
ret = gb_pm_runtime_get_sync(bundle);
|
if (ret < 0)
|
return ret;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
req.state = state ? 1 : 0;
|
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_SET_FLASH_STROBE,
|
&req, sizeof(req), NULL, 0);
|
if (!ret)
|
channel->strobe_state = state;
|
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
return ret;
|
}
|
|
static int gb_lights_flash_strobe_get(struct led_classdev_flash *fcdev,
|
bool *state)
|
{
|
struct gb_channel *channel = container_of(fcdev, struct gb_channel,
|
fled);
|
|
*state = channel->strobe_state;
|
return 0;
|
}
|
|
static int gb_lights_flash_timeout_set(struct led_classdev_flash *fcdev,
|
u32 timeout)
|
{
|
struct gb_channel *channel = container_of(fcdev, struct gb_channel,
|
fled);
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_bundle *bundle = connection->bundle;
|
struct gb_lights_set_flash_timeout_request req;
|
int ret;
|
|
if (channel->releasing)
|
return -ESHUTDOWN;
|
|
ret = gb_pm_runtime_get_sync(bundle);
|
if (ret < 0)
|
return ret;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
req.timeout_us = cpu_to_le32(timeout);
|
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_SET_FLASH_TIMEOUT,
|
&req, sizeof(req), NULL, 0);
|
if (!ret)
|
fcdev->timeout.val = timeout;
|
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
return ret;
|
}
|
|
static int gb_lights_flash_fault_get(struct led_classdev_flash *fcdev,
|
u32 *fault)
|
{
|
struct gb_channel *channel = container_of(fcdev, struct gb_channel,
|
fled);
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_bundle *bundle = connection->bundle;
|
struct gb_lights_get_flash_fault_request req;
|
struct gb_lights_get_flash_fault_response resp;
|
int ret;
|
|
if (channel->releasing)
|
return -ESHUTDOWN;
|
|
ret = gb_pm_runtime_get_sync(bundle);
|
if (ret < 0)
|
return ret;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_GET_FLASH_FAULT,
|
&req, sizeof(req), &resp, sizeof(resp));
|
if (!ret)
|
*fault = le32_to_cpu(resp.fault);
|
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
return ret;
|
}
|
|
static const struct led_flash_ops gb_lights_flash_ops = {
|
.flash_brightness_set = gb_lights_flash_intensity_set,
|
.flash_brightness_get = gb_lights_flash_intensity_get,
|
.strobe_set = gb_lights_flash_strobe_set,
|
.strobe_get = gb_lights_flash_strobe_get,
|
.timeout_set = gb_lights_flash_timeout_set,
|
.fault_get = gb_lights_flash_fault_get,
|
};
|
|
static int __gb_lights_channel_torch_attach(struct gb_channel *channel,
|
struct gb_channel *channel_torch)
|
{
|
char *name;
|
|
/* we can only attach torch to a flash channel */
|
if (!(channel->mode & GB_CHANNEL_MODE_FLASH))
|
return 0;
|
|
/* Move torch brightness to the destination */
|
channel->led->max_brightness = channel_torch->led->max_brightness;
|
|
/* append mode name to flash name */
|
name = kasprintf(GFP_KERNEL, "%s_%s", channel->led->name,
|
channel_torch->mode_name);
|
if (!name)
|
return -ENOMEM;
|
kfree(channel->led->name);
|
channel->led->name = name;
|
|
channel_torch->led = channel->led;
|
|
return 0;
|
}
|
|
static int __gb_lights_flash_led_register(struct gb_channel *channel)
|
{
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct led_classdev_flash *fled = &channel->fled;
|
struct led_flash_setting *fset;
|
struct gb_channel *channel_torch;
|
int ret;
|
|
fled->ops = &gb_lights_flash_ops;
|
|
fled->led_cdev.flags |= LED_DEV_CAP_FLASH;
|
|
fset = &fled->brightness;
|
fset->min = channel->intensity_uA.min;
|
fset->max = channel->intensity_uA.max;
|
fset->step = channel->intensity_uA.step;
|
fset->val = channel->intensity_uA.max;
|
|
/* Only the flash mode have the timeout constraints settings */
|
if (channel->mode & GB_CHANNEL_MODE_FLASH) {
|
fset = &fled->timeout;
|
fset->min = channel->timeout_us.min;
|
fset->max = channel->timeout_us.max;
|
fset->step = channel->timeout_us.step;
|
fset->val = channel->timeout_us.max;
|
}
|
|
/*
|
* If light have torch mode channel, this channel will be the led
|
* classdev of the registered above flash classdev
|
*/
|
channel_torch = get_channel_from_mode(channel->light,
|
GB_CHANNEL_MODE_TORCH);
|
if (channel_torch) {
|
ret = __gb_lights_channel_torch_attach(channel, channel_torch);
|
if (ret < 0)
|
goto fail;
|
}
|
|
ret = led_classdev_flash_register(&connection->bundle->dev, fled);
|
if (ret < 0)
|
goto fail;
|
|
channel->is_registered = true;
|
return 0;
|
fail:
|
channel->led = NULL;
|
return ret;
|
}
|
|
static void __gb_lights_flash_led_unregister(struct gb_channel *channel)
|
{
|
if (!channel->is_registered)
|
return;
|
|
led_classdev_flash_unregister(&channel->fled);
|
}
|
|
static int gb_lights_channel_flash_config(struct gb_channel *channel)
|
{
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct gb_lights_get_channel_flash_config_request req;
|
struct gb_lights_get_channel_flash_config_response conf;
|
struct led_flash_setting *fset;
|
int ret;
|
|
req.light_id = channel->light->id;
|
req.channel_id = channel->id;
|
|
ret = gb_operation_sync(connection,
|
GB_LIGHTS_TYPE_GET_CHANNEL_FLASH_CONFIG,
|
&req, sizeof(req), &conf, sizeof(conf));
|
if (ret < 0)
|
return ret;
|
|
/*
|
* Intensity constraints for flash related modes: flash, torch,
|
* indicator. They will be needed for v4l2 registration.
|
*/
|
fset = &channel->intensity_uA;
|
fset->min = le32_to_cpu(conf.intensity_min_uA);
|
fset->max = le32_to_cpu(conf.intensity_max_uA);
|
fset->step = le32_to_cpu(conf.intensity_step_uA);
|
|
/*
|
* On flash type, max brightness is set as the number of intensity steps
|
* available.
|
*/
|
channel->led->max_brightness = (fset->max - fset->min) / fset->step;
|
|
/* Only the flash mode have the timeout constraints settings */
|
if (channel->mode & GB_CHANNEL_MODE_FLASH) {
|
fset = &channel->timeout_us;
|
fset->min = le32_to_cpu(conf.timeout_min_us);
|
fset->max = le32_to_cpu(conf.timeout_max_us);
|
fset->step = le32_to_cpu(conf.timeout_step_us);
|
}
|
|
return 0;
|
}
|
#else
|
static int gb_lights_channel_flash_config(struct gb_channel *channel)
|
{
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
|
dev_err(&connection->bundle->dev, "no support for flash devices\n");
|
return 0;
|
}
|
|
static int __gb_lights_flash_led_register(struct gb_channel *channel)
|
{
|
return 0;
|
}
|
|
static void __gb_lights_flash_led_unregister(struct gb_channel *channel)
|
{
|
}
|
|
#endif
|
|
static int __gb_lights_led_register(struct gb_channel *channel)
|
{
|
struct gb_connection *connection = get_conn_from_channel(channel);
|
struct led_classdev *cdev = get_channel_cdev(channel);
|
int ret;
|
|
ret = led_classdev_register(&connection->bundle->dev, cdev);
|
if (ret < 0)
|
channel->led = NULL;
|
else
|
channel->is_registered = true;
|
return ret;
|
}
|
|
static int gb_lights_channel_register(struct gb_channel *channel)
|
{
|
/* Normal LED channel, just register in led classdev and we are done */
|
if (!is_channel_flash(channel))
|
return __gb_lights_led_register(channel);
|
|
/*
|
* Flash Type need more work, register flash classdev, indicator as
|
* flash classdev, torch will be led classdev of the flash classdev.
|
*/
|
if (!(channel->mode & GB_CHANNEL_MODE_TORCH))
|
return __gb_lights_flash_led_register(channel);
|
|
return 0;
|
}
|
|
static void __gb_lights_led_unregister(struct gb_channel *channel)
|
{
|
struct led_classdev *cdev = get_channel_cdev(channel);
|
|
if (!channel->is_registered)
|
return;
|
|
led_classdev_unregister(cdev);
|
kfree(cdev->name);
|
cdev->name = NULL;
|
channel->led = NULL;
|
}
|
|
static void gb_lights_channel_unregister(struct gb_channel *channel)
|
{
|
/* The same as register, handle channels differently */
|
if (!is_channel_flash(channel)) {
|
__gb_lights_led_unregister(channel);
|
return;
|
}
|
|
if (channel->mode & GB_CHANNEL_MODE_TORCH)
|
__gb_lights_led_unregister(channel);
|
else
|
__gb_lights_flash_led_unregister(channel);
|
}
|
|
static int gb_lights_channel_config(struct gb_light *light,
|
struct gb_channel *channel)
|
{
|
struct gb_lights_get_channel_config_response conf;
|
struct gb_lights_get_channel_config_request req;
|
struct gb_connection *connection = get_conn_from_light(light);
|
struct led_classdev *cdev = get_channel_cdev(channel);
|
char *name;
|
int ret;
|
|
req.light_id = light->id;
|
req.channel_id = channel->id;
|
|
ret = gb_operation_sync(connection, GB_LIGHTS_TYPE_GET_CHANNEL_CONFIG,
|
&req, sizeof(req), &conf, sizeof(conf));
|
if (ret < 0)
|
return ret;
|
|
channel->light = light;
|
channel->mode = le32_to_cpu(conf.mode);
|
channel->flags = le32_to_cpu(conf.flags);
|
channel->color = le32_to_cpu(conf.color);
|
channel->color_name = kstrndup(conf.color_name, NAMES_MAX, GFP_KERNEL);
|
if (!channel->color_name)
|
return -ENOMEM;
|
channel->mode_name = kstrndup(conf.mode_name, NAMES_MAX, GFP_KERNEL);
|
if (!channel->mode_name)
|
return -ENOMEM;
|
|
channel->led = cdev;
|
|
name = kasprintf(GFP_KERNEL, "%s:%s:%s", light->name,
|
channel->color_name, channel->mode_name);
|
if (!name)
|
return -ENOMEM;
|
|
cdev->name = name;
|
|
cdev->max_brightness = conf.max_brightness;
|
|
ret = channel_attr_groups_set(channel, cdev);
|
if (ret < 0)
|
return ret;
|
|
gb_lights_led_operations_set(channel, cdev);
|
|
/*
|
* If it is not a flash related channel (flash, torch or indicator) we
|
* are done here. If not, continue and fetch flash related
|
* configurations.
|
*/
|
if (!is_channel_flash(channel))
|
return ret;
|
|
light->has_flash = true;
|
|
return gb_lights_channel_flash_config(channel);
|
}
|
|
static int gb_lights_light_config(struct gb_lights *glights, u8 id)
|
{
|
struct gb_light *light = &glights->lights[id];
|
struct gb_lights_get_light_config_request req;
|
struct gb_lights_get_light_config_response conf;
|
int ret;
|
int i;
|
|
light->glights = glights;
|
light->id = id;
|
|
req.id = id;
|
|
ret = gb_operation_sync(glights->connection,
|
GB_LIGHTS_TYPE_GET_LIGHT_CONFIG,
|
&req, sizeof(req), &conf, sizeof(conf));
|
if (ret < 0)
|
return ret;
|
|
if (!conf.channel_count)
|
return -EINVAL;
|
if (!strlen(conf.name))
|
return -EINVAL;
|
|
light->channels_count = conf.channel_count;
|
light->name = kstrndup(conf.name, NAMES_MAX, GFP_KERNEL);
|
if (!light->name)
|
return -ENOMEM;
|
light->channels = kcalloc(light->channels_count,
|
sizeof(struct gb_channel), GFP_KERNEL);
|
if (!light->channels)
|
return -ENOMEM;
|
|
/* First we collect all the configurations for all channels */
|
for (i = 0; i < light->channels_count; i++) {
|
light->channels[i].id = i;
|
ret = gb_lights_channel_config(light, &light->channels[i]);
|
if (ret < 0)
|
return ret;
|
}
|
|
return 0;
|
}
|
|
static int gb_lights_light_register(struct gb_light *light)
|
{
|
int ret;
|
int i;
|
|
/*
|
* Then, if everything went ok in getting configurations, we register
|
* the classdev, flash classdev and v4l2 subsystem, if a flash device is
|
* found.
|
*/
|
for (i = 0; i < light->channels_count; i++) {
|
ret = gb_lights_channel_register(&light->channels[i]);
|
if (ret < 0)
|
return ret;
|
|
mutex_init(&light->channels[i].lock);
|
}
|
|
light->ready = true;
|
|
if (light->has_flash) {
|
ret = gb_lights_light_v4l2_register(light);
|
if (ret < 0) {
|
light->has_flash = false;
|
return ret;
|
}
|
}
|
|
return 0;
|
}
|
|
static void gb_lights_channel_free(struct gb_channel *channel)
|
{
|
kfree(channel->attrs);
|
kfree(channel->attr_group);
|
kfree(channel->attr_groups);
|
kfree(channel->color_name);
|
kfree(channel->mode_name);
|
mutex_destroy(&channel->lock);
|
}
|
|
static void gb_lights_channel_release(struct gb_channel *channel)
|
{
|
channel->releasing = true;
|
|
gb_lights_channel_unregister(channel);
|
|
gb_lights_channel_free(channel);
|
}
|
|
static void gb_lights_light_release(struct gb_light *light)
|
{
|
int i;
|
|
light->ready = false;
|
|
if (light->has_flash)
|
gb_lights_light_v4l2_unregister(light);
|
light->has_flash = false;
|
|
for (i = 0; i < light->channels_count; i++)
|
gb_lights_channel_release(&light->channels[i]);
|
light->channels_count = 0;
|
|
kfree(light->channels);
|
light->channels = NULL;
|
kfree(light->name);
|
light->name = NULL;
|
}
|
|
static void gb_lights_release(struct gb_lights *glights)
|
{
|
int i;
|
|
if (!glights)
|
return;
|
|
mutex_lock(&glights->lights_lock);
|
if (!glights->lights)
|
goto free_glights;
|
|
for (i = 0; i < glights->lights_count; i++)
|
gb_lights_light_release(&glights->lights[i]);
|
|
kfree(glights->lights);
|
|
free_glights:
|
mutex_unlock(&glights->lights_lock);
|
mutex_destroy(&glights->lights_lock);
|
kfree(glights);
|
}
|
|
static int gb_lights_get_count(struct gb_lights *glights)
|
{
|
struct gb_lights_get_lights_response resp;
|
int ret;
|
|
ret = gb_operation_sync(glights->connection, GB_LIGHTS_TYPE_GET_LIGHTS,
|
NULL, 0, &resp, sizeof(resp));
|
if (ret < 0)
|
return ret;
|
|
if (!resp.lights_count)
|
return -EINVAL;
|
|
glights->lights_count = resp.lights_count;
|
|
return 0;
|
}
|
|
static int gb_lights_create_all(struct gb_lights *glights)
|
{
|
struct gb_connection *connection = glights->connection;
|
int ret;
|
int i;
|
|
mutex_lock(&glights->lights_lock);
|
ret = gb_lights_get_count(glights);
|
if (ret < 0)
|
goto out;
|
|
glights->lights = kcalloc(glights->lights_count,
|
sizeof(struct gb_light), GFP_KERNEL);
|
if (!glights->lights) {
|
ret = -ENOMEM;
|
goto out;
|
}
|
|
for (i = 0; i < glights->lights_count; i++) {
|
ret = gb_lights_light_config(glights, i);
|
if (ret < 0) {
|
dev_err(&connection->bundle->dev,
|
"Fail to configure lights device\n");
|
goto out;
|
}
|
}
|
|
out:
|
mutex_unlock(&glights->lights_lock);
|
return ret;
|
}
|
|
static int gb_lights_register_all(struct gb_lights *glights)
|
{
|
struct gb_connection *connection = glights->connection;
|
int ret = 0;
|
int i;
|
|
mutex_lock(&glights->lights_lock);
|
for (i = 0; i < glights->lights_count; i++) {
|
ret = gb_lights_light_register(&glights->lights[i]);
|
if (ret < 0) {
|
dev_err(&connection->bundle->dev,
|
"Fail to enable lights device\n");
|
break;
|
}
|
}
|
|
mutex_unlock(&glights->lights_lock);
|
return ret;
|
}
|
|
static int gb_lights_request_handler(struct gb_operation *op)
|
{
|
struct gb_connection *connection = op->connection;
|
struct device *dev = &connection->bundle->dev;
|
struct gb_lights *glights = gb_connection_get_data(connection);
|
struct gb_light *light;
|
struct gb_message *request;
|
struct gb_lights_event_request *payload;
|
int ret = 0;
|
u8 light_id;
|
u8 event;
|
|
if (op->type != GB_LIGHTS_TYPE_EVENT) {
|
dev_err(dev, "Unsupported unsolicited event: %u\n", op->type);
|
return -EINVAL;
|
}
|
|
request = op->request;
|
|
if (request->payload_size < sizeof(*payload)) {
|
dev_err(dev, "Wrong event size received (%zu < %zu)\n",
|
request->payload_size, sizeof(*payload));
|
return -EINVAL;
|
}
|
|
payload = request->payload;
|
light_id = payload->light_id;
|
|
if (light_id >= glights->lights_count ||
|
!glights->lights[light_id].ready) {
|
dev_err(dev, "Event received for unconfigured light id: %d\n",
|
light_id);
|
return -EINVAL;
|
}
|
|
event = payload->event;
|
|
if (event & GB_LIGHTS_LIGHT_CONFIG) {
|
light = &glights->lights[light_id];
|
|
mutex_lock(&glights->lights_lock);
|
gb_lights_light_release(light);
|
ret = gb_lights_light_config(glights, light_id);
|
if (!ret)
|
ret = gb_lights_light_register(light);
|
if (ret < 0)
|
gb_lights_light_release(light);
|
mutex_unlock(&glights->lights_lock);
|
}
|
|
return ret;
|
}
|
|
static int gb_lights_probe(struct gb_bundle *bundle,
|
const struct greybus_bundle_id *id)
|
{
|
struct greybus_descriptor_cport *cport_desc;
|
struct gb_connection *connection;
|
struct gb_lights *glights;
|
int ret;
|
|
if (bundle->num_cports != 1)
|
return -ENODEV;
|
|
cport_desc = &bundle->cport_desc[0];
|
if (cport_desc->protocol_id != GREYBUS_PROTOCOL_LIGHTS)
|
return -ENODEV;
|
|
glights = kzalloc(sizeof(*glights), GFP_KERNEL);
|
if (!glights)
|
return -ENOMEM;
|
|
mutex_init(&glights->lights_lock);
|
|
connection = gb_connection_create(bundle, le16_to_cpu(cport_desc->id),
|
gb_lights_request_handler);
|
if (IS_ERR(connection)) {
|
ret = PTR_ERR(connection);
|
goto out;
|
}
|
|
glights->connection = connection;
|
gb_connection_set_data(connection, glights);
|
|
greybus_set_drvdata(bundle, glights);
|
|
/* We aren't ready to receive an incoming request yet */
|
ret = gb_connection_enable_tx(connection);
|
if (ret)
|
goto error_connection_destroy;
|
|
/*
|
* Setup all the lights devices over this connection, if anything goes
|
* wrong tear down all lights
|
*/
|
ret = gb_lights_create_all(glights);
|
if (ret < 0)
|
goto error_connection_disable;
|
|
/* We are ready to receive an incoming request now, enable RX as well */
|
ret = gb_connection_enable(connection);
|
if (ret)
|
goto error_connection_disable;
|
|
/* Enable & register lights */
|
ret = gb_lights_register_all(glights);
|
if (ret < 0)
|
goto error_connection_disable;
|
|
gb_pm_runtime_put_autosuspend(bundle);
|
|
return 0;
|
|
error_connection_disable:
|
gb_connection_disable(connection);
|
error_connection_destroy:
|
gb_connection_destroy(connection);
|
out:
|
gb_lights_release(glights);
|
return ret;
|
}
|
|
static void gb_lights_disconnect(struct gb_bundle *bundle)
|
{
|
struct gb_lights *glights = greybus_get_drvdata(bundle);
|
|
if (gb_pm_runtime_get_sync(bundle))
|
gb_pm_runtime_get_noresume(bundle);
|
|
gb_connection_disable(glights->connection);
|
gb_connection_destroy(glights->connection);
|
|
gb_lights_release(glights);
|
}
|
|
static const struct greybus_bundle_id gb_lights_id_table[] = {
|
{ GREYBUS_DEVICE_CLASS(GREYBUS_CLASS_LIGHTS) },
|
{ }
|
};
|
MODULE_DEVICE_TABLE(greybus, gb_lights_id_table);
|
|
static struct greybus_driver gb_lights_driver = {
|
.name = "lights",
|
.probe = gb_lights_probe,
|
.disconnect = gb_lights_disconnect,
|
.id_table = gb_lights_id_table,
|
};
|
module_greybus_driver(gb_lights_driver);
|
|
MODULE_LICENSE("GPL v2");
|