// SPDX-License-Identifier: GPL-2.0 /* Author: Hans de Goede */ #include #include #include #include "common.h" static int int3472_pled_set(struct led_classdev *led_cdev, enum led_brightness brightness) { struct int3472_discrete_device *int3472 = container_of(led_cdev, struct int3472_discrete_device, pled.classdev); gpiod_set_value_cansleep(int3472->pled.gpio, brightness); return 0; } int skl_int3472_register_pled(struct int3472_discrete_device *int3472, struct acpi_resource_gpio *agpio, u32 polarity) { char *p, *path = agpio->resource_source.string_ptr; int ret; if (int3472->pled.classdev.dev) return -EBUSY; int3472->pled.gpio = acpi_get_and_request_gpiod(path, agpio->pin_table[0], "int3472,privacy-led"); if (IS_ERR(int3472->pled.gpio)) return dev_err_probe(int3472->dev, PTR_ERR(int3472->pled.gpio), "getting privacy LED GPIO\n"); if (polarity == GPIO_ACTIVE_LOW) gpiod_toggle_active_low(int3472->pled.gpio); /* Ensure the pin is in output mode and non-active state */ gpiod_direction_output(int3472->pled.gpio, 0); /* Generate the name, replacing the ':' in the ACPI devname with '_' */ snprintf(int3472->pled.name, sizeof(int3472->pled.name), "%s::privacy_led", acpi_dev_name(int3472->sensor)); p = strchr(int3472->pled.name, ':'); if (p) *p = '_'; int3472->pled.classdev.name = int3472->pled.name; int3472->pled.classdev.max_brightness = 1; int3472->pled.classdev.brightness_set_blocking = int3472_pled_set; ret = led_classdev_register(int3472->dev, &int3472->pled.classdev); if (ret) goto err_free_gpio; int3472->pled.lookup.provider = int3472->pled.name; int3472->pled.lookup.dev_id = int3472->sensor_name; int3472->pled.lookup.con_id = "privacy-led"; led_add_lookup(&int3472->pled.lookup); return 0; err_free_gpio: gpiod_put(int3472->pled.gpio); return ret; } void skl_int3472_unregister_pled(struct int3472_discrete_device *int3472) { if (IS_ERR_OR_NULL(int3472->pled.classdev.dev)) return; led_remove_lookup(&int3472->pled.lookup); led_classdev_unregister(&int3472->pled.classdev); gpiod_put(int3472->pled.gpio); }