1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
|
/* Copyright 2021 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include <device.h>
#include <devicetree.h>
#include <drivers/pwm.h>
#include <logging/log.h>
#include "common.h"
#include "console.h"
#include "drivers/cros_displight.h"
#include "ec_commands.h"
#include "host_command.h"
#include "pwm.h"
#include "keyboard_backlight.h"
#include "util.h"
LOG_MODULE_REGISTER(pwm_shim, LOG_LEVEL_ERR);
#define PWM_RAW_TO_PERCENT(v) \
DIV_ROUND_NEAREST((uint32_t)(v) * 100, UINT16_MAX)
#define PWM_PERCENT_TO_RAW(v) ((uint32_t)(v) * UINT16_MAX / 100)
#define HAS_PWM_GENERIC_CHANNEL(compat) \
DT_NODE_HAS_PROP(DT_COMPAT_GET_ANY_STATUS_OKAY(compat), \
generic_pwm_channel)
#define PWM_GENERIC_CHANNEL_ID(compat) \
DT_PROP(DT_COMPAT_GET_ANY_STATUS_OKAY(compat), \
generic_pwm_channel)
#ifdef CONFIG_PWM_KBLIGHT
static bool pwm_is_kblight(int type, int index)
{
if (type == EC_PWM_TYPE_KB_LIGHT)
return true;
#if HAS_PWM_GENERIC_CHANNEL(cros_ec_kblight_pwm)
if (type == EC_PWM_TYPE_GENERIC &&
index == PWM_GENERIC_CHANNEL_ID(cros_ec_kblight_pwm))
return true;
#endif /* HAS_PWM_GENERIC_CHANNEL(cros_ec_kblight_pwm) */
return false;
}
#endif /* CONFIG_PWM_KBLIGHT */
#ifdef CONFIG_PLATFORM_EC_PWM_DISPLIGHT
static bool pwm_is_displight(int type, int index)
{
if (type == EC_PWM_TYPE_DISPLAY_LIGHT)
return true;
#if HAS_PWM_GENERIC_CHANNEL(cros_ec_displight)
if (type == EC_PWM_TYPE_GENERIC &&
index == PWM_GENERIC_CHANNEL_ID(cros_ec_displight))
return true;
#endif /* HAS_PWM_GENERIC_CHANNEL(cros_ec_displight) */
return false;
}
#endif /* CONFIG_PLATFORM_EC_PWM_DISPLIGHT */
static enum ec_status host_command_pwm_set_duty(
struct host_cmd_handler_args *args)
{
__maybe_unused const struct ec_params_pwm_set_duty *p = args->params;
#ifdef CONFIG_PLATFORM_EC_PWM_KBLIGHT
if (pwm_is_kblight(p->pwm_type, p->index)) {
kblight_set(PWM_RAW_TO_PERCENT(p->duty));
kblight_enable(p->duty > 0);
return EC_RES_SUCCESS;
}
#endif
#ifdef CONFIG_PLATFORM_EC_PWM_DISPLIGHT
if (pwm_is_displight(p->pwm_type, p->index)) {
displight_set(PWM_RAW_TO_PERCENT(p->duty));
return EC_RES_SUCCESS;
}
#endif
return EC_RES_INVALID_PARAM;
}
DECLARE_HOST_COMMAND(EC_CMD_PWM_SET_DUTY,
host_command_pwm_set_duty,
EC_VER_MASK(0));
static enum ec_status host_command_pwm_get_duty(
struct host_cmd_handler_args *args)
{
__maybe_unused const struct ec_params_pwm_get_duty *p = args->params;
__maybe_unused struct ec_response_pwm_get_duty *r = args->response;
#ifdef CONFIG_PLATFORM_EC_PWM_KBLIGHT
if (pwm_is_kblight(p->pwm_type, p->index)) {
r->duty = PWM_PERCENT_TO_RAW(kblight_get());
args->response_size = sizeof(*r);
return EC_RES_SUCCESS;
}
#endif
#ifdef CONFIG_PLATFORM_EC_PWM_DISPLIGHT
if (pwm_is_displight(p->pwm_type, p->index)) {
r->duty = PWM_PERCENT_TO_RAW(displight_get());
args->response_size = sizeof(*r);
return EC_RES_SUCCESS;
}
#endif
return EC_RES_INVALID_PARAM;
}
DECLARE_HOST_COMMAND(EC_CMD_PWM_GET_DUTY,
host_command_pwm_get_duty,
EC_VER_MASK(0));
|