1d4afb5ceSopenharmony_ci/*
2d4afb5ceSopenharmony_ci * Generic GPIO led
3d4afb5ceSopenharmony_ci *
4d4afb5ceSopenharmony_ci * Copyright (C) 2019 - 2020 Andy Green <andy@warmcat.com>
5d4afb5ceSopenharmony_ci *
6d4afb5ceSopenharmony_ci * Permission is hereby granted, free of charge, to any person obtaining a copy
7d4afb5ceSopenharmony_ci * of this software and associated documentation files (the "Software"), to
8d4afb5ceSopenharmony_ci * deal in the Software without restriction, including without limitation the
9d4afb5ceSopenharmony_ci * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10d4afb5ceSopenharmony_ci * sell copies of the Software, and to permit persons to whom the Software is
11d4afb5ceSopenharmony_ci * furnished to do so, subject to the following conditions:
12d4afb5ceSopenharmony_ci *
13d4afb5ceSopenharmony_ci * The above copyright notice and this permission notice shall be included in
14d4afb5ceSopenharmony_ci * all copies or substantial portions of the Software.
15d4afb5ceSopenharmony_ci *
16d4afb5ceSopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17d4afb5ceSopenharmony_ci * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18d4afb5ceSopenharmony_ci * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19d4afb5ceSopenharmony_ci * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20d4afb5ceSopenharmony_ci * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21d4afb5ceSopenharmony_ci * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22d4afb5ceSopenharmony_ci * IN THE SOFTWARE.
23d4afb5ceSopenharmony_ci */
24d4afb5ceSopenharmony_ci#include "private-lib-core.h"
25d4afb5ceSopenharmony_ci
26d4afb5ceSopenharmony_ci#include "drivers/led/private-lib-drivers-led.h"
27d4afb5ceSopenharmony_ci
28d4afb5ceSopenharmony_ci/*
29d4afb5ceSopenharmony_ci * 64 entry interpolated CIE correction
30d4afb5ceSopenharmony_ci * https://en.wikipedia.org/wiki/Lightness
31d4afb5ceSopenharmony_ci */
32d4afb5ceSopenharmony_ci
33d4afb5ceSopenharmony_ciuint16_t cie[] = {
34d4afb5ceSopenharmony_ci	0, 113, 227, 340, 454, 568, 688, 824, 976, 1146,
35d4afb5ceSopenharmony_ci	1335, 1543, 1772, 2023, 2296, 2592, 2914, 3260, 3633, 4034,
36d4afb5ceSopenharmony_ci	4463, 4921, 5409, 5929, 6482, 7067, 7687, 8341, 9032, 9761,
37d4afb5ceSopenharmony_ci	10527, 11332, 12178, 13064, 13993, 14964, 15980, 17040, 18146, 19299,
38d4afb5ceSopenharmony_ci	20500, 21750, 23049, 24400, 25802, 27256, 28765, 30328, 31946, 33622,
39d4afb5ceSopenharmony_ci	35354, 37146, 38996, 40908, 42881, 44916, 47014, 49177, 51406, 53700,
40d4afb5ceSopenharmony_ci	56062, 58492, 60992, 63561,
41d4afb5ceSopenharmony_ci	65535 /* for interpolation */
42d4afb5ceSopenharmony_ci};
43d4afb5ceSopenharmony_ci
44d4afb5ceSopenharmony_ci/*
45d4afb5ceSopenharmony_ci * This is the default intensity correction function, it can be overridden
46d4afb5ceSopenharmony_ci * per-led to eg, normalize intensity of different leds
47d4afb5ceSopenharmony_ci */
48d4afb5ceSopenharmony_ci
49d4afb5ceSopenharmony_cistatic lws_led_intensity_t
50d4afb5ceSopenharmony_cicie_antilog(lws_led_intensity_t lin)
51d4afb5ceSopenharmony_ci{
52d4afb5ceSopenharmony_ci        return (cie[lin >> 10]       * (0x3ff - (lin & 0x3ff)) +
53d4afb5ceSopenharmony_ci        	cie[(lin >> 10) + 1] * (lin & 0x3ff)) / 0x3ff;
54d4afb5ceSopenharmony_ci}
55d4afb5ceSopenharmony_ci
56d4afb5ceSopenharmony_cistatic void
57d4afb5ceSopenharmony_cilws_seq_advance(lws_led_state_t *lcs, lws_led_state_ch_t *ch)
58d4afb5ceSopenharmony_ci{
59d4afb5ceSopenharmony_ci	if (!ch->seq)
60d4afb5ceSopenharmony_ci		return;
61d4afb5ceSopenharmony_ci
62d4afb5ceSopenharmony_ci	if (ch->phase_budget != LWS_SEQ_LEDPHASE_TOTAL_ENDLESS &&
63d4afb5ceSopenharmony_ci	    (ch->phase_budget < ch->step || !ch->phase_budget)) {
64d4afb5ceSopenharmony_ci
65d4afb5ceSopenharmony_ci		/* we are done */
66d4afb5ceSopenharmony_ci
67d4afb5ceSopenharmony_ci		ch->seq = NULL;
68d4afb5ceSopenharmony_ci		if (!(--lcs->timer_refcount)) {
69d4afb5ceSopenharmony_ci#if defined(LWS_PLAT_TIMER_STOP)
70d4afb5ceSopenharmony_ci			LWS_PLAT_TIMER_STOP(lcs->timer);
71d4afb5ceSopenharmony_ci#endif
72d4afb5ceSopenharmony_ci		}
73d4afb5ceSopenharmony_ci
74d4afb5ceSopenharmony_ci		return;
75d4afb5ceSopenharmony_ci	}
76d4afb5ceSopenharmony_ci
77d4afb5ceSopenharmony_ci	ch->ph += ch->step;
78d4afb5ceSopenharmony_ci	if (ch->phase_budget != LWS_SEQ_LEDPHASE_TOTAL_ENDLESS)
79d4afb5ceSopenharmony_ci		ch->phase_budget -= ch->step;
80d4afb5ceSopenharmony_ci}
81d4afb5ceSopenharmony_ci
82d4afb5ceSopenharmony_cistatic lws_led_intensity_t
83d4afb5ceSopenharmony_cilws_seq_sample(const lws_led_gpio_map_t *map, lws_led_state_chs_t *chs)
84d4afb5ceSopenharmony_ci{
85d4afb5ceSopenharmony_ci	unsigned int i;
86d4afb5ceSopenharmony_ci
87d4afb5ceSopenharmony_ci	if (chs->seqs[LLSI_CURR].seq)
88d4afb5ceSopenharmony_ci		chs->seqs[LLSI_CURR].last = chs->seqs[LLSI_CURR].seq->
89d4afb5ceSopenharmony_ci						func(chs->seqs[LLSI_CURR].ph);
90d4afb5ceSopenharmony_ci
91d4afb5ceSopenharmony_ci	if (chs->seqs[LLSI_TRANS].seq) {
92d4afb5ceSopenharmony_ci		/*
93d4afb5ceSopenharmony_ci		 * If a transition is ongoing, we need to use the transition
94d4afb5ceSopenharmony_ci		 * intensity as the mixing factor between the still-live current
95d4afb5ceSopenharmony_ci		 * and newly-live next sequences
96d4afb5ceSopenharmony_ci		 */
97d4afb5ceSopenharmony_ci		chs->seqs[LLSI_TRANS].last = chs->seqs[LLSI_TRANS].seq->
98d4afb5ceSopenharmony_ci						func(chs->seqs[LLSI_TRANS].ph);
99d4afb5ceSopenharmony_ci
100d4afb5ceSopenharmony_ci		if (chs->seqs[LLSI_NEXT].seq)
101d4afb5ceSopenharmony_ci			chs->seqs[LLSI_NEXT].last = chs->seqs[LLSI_NEXT].seq->
102d4afb5ceSopenharmony_ci						  func(chs->seqs[LLSI_NEXT].ph);
103d4afb5ceSopenharmony_ci
104d4afb5ceSopenharmony_ci		i = (lws_led_intensity_t)(((
105d4afb5ceSopenharmony_ci				(unsigned int)chs->seqs[LLSI_CURR].last *
106d4afb5ceSopenharmony_ci				(65535 - chs->seqs[LLSI_TRANS].last) >> 16) +
107d4afb5ceSopenharmony_ci			(((unsigned int)chs->seqs[LLSI_NEXT].last *
108d4afb5ceSopenharmony_ci			 (unsigned int)chs->seqs[LLSI_TRANS].last) >> 16)));
109d4afb5ceSopenharmony_ci	} else
110d4afb5ceSopenharmony_ci		i = chs->seqs[LLSI_CURR].last;
111d4afb5ceSopenharmony_ci
112d4afb5ceSopenharmony_ci	return map->intensity_correction ? map->intensity_correction(i) :
113d4afb5ceSopenharmony_ci					   cie_antilog((lws_led_intensity_t)i);
114d4afb5ceSopenharmony_ci}
115d4afb5ceSopenharmony_ci
116d4afb5ceSopenharmony_civoid
117d4afb5ceSopenharmony_cilws_seq_timer_handle(lws_led_state_t *lcs)
118d4afb5ceSopenharmony_ci{
119d4afb5ceSopenharmony_ci	lws_led_gpio_controller_t *lgc = lcs->controller;
120d4afb5ceSopenharmony_ci	lws_led_state_chs_t *chs = (lws_led_state_chs_t *)&lcs[1];
121d4afb5ceSopenharmony_ci	const lws_led_gpio_map_t *map = &lgc->led_map[0];
122d4afb5ceSopenharmony_ci	unsigned int n;
123d4afb5ceSopenharmony_ci
124d4afb5ceSopenharmony_ci	for (n = 0; n < lgc->count_leds; n++) {
125d4afb5ceSopenharmony_ci
126d4afb5ceSopenharmony_ci		lgc->led_ops.intensity(&lgc->led_ops, map->name,
127d4afb5ceSopenharmony_ci				       lws_seq_sample(map, chs));
128d4afb5ceSopenharmony_ci
129d4afb5ceSopenharmony_ci		lws_seq_advance(lcs, &chs->seqs[LLSI_CURR]);
130d4afb5ceSopenharmony_ci
131d4afb5ceSopenharmony_ci		if (chs->seqs[LLSI_TRANS].seq) {
132d4afb5ceSopenharmony_ci			lws_seq_advance(lcs, &chs->seqs[LLSI_NEXT]);
133d4afb5ceSopenharmony_ci			lws_seq_advance(lcs, &chs->seqs[LLSI_TRANS]);
134d4afb5ceSopenharmony_ci
135d4afb5ceSopenharmony_ci			/*
136d4afb5ceSopenharmony_ci			 * When we finished the transition, we can make the
137d4afb5ceSopenharmony_ci			 * "next" sequence the current sequence and no need for
138d4afb5ceSopenharmony_ci			 * a "next" or a transition any more.
139d4afb5ceSopenharmony_ci			 */
140d4afb5ceSopenharmony_ci
141d4afb5ceSopenharmony_ci			if (!chs->seqs[LLSI_TRANS].seq) {
142d4afb5ceSopenharmony_ci				chs->seqs[LLSI_CURR] = chs->seqs[LLSI_NEXT];
143d4afb5ceSopenharmony_ci				chs->seqs[LLSI_NEXT].seq = NULL;
144d4afb5ceSopenharmony_ci			}
145d4afb5ceSopenharmony_ci		}
146d4afb5ceSopenharmony_ci
147d4afb5ceSopenharmony_ci		map++;
148d4afb5ceSopenharmony_ci		chs++;
149d4afb5ceSopenharmony_ci	}
150d4afb5ceSopenharmony_ci}
151d4afb5ceSopenharmony_ci
152d4afb5ceSopenharmony_cistatic int
153d4afb5ceSopenharmony_cilws_led_set_chs_seq(struct lws_led_state *lcs, lws_led_state_ch_t *dest,
154d4afb5ceSopenharmony_ci		    const lws_led_sequence_def_t *def)
155d4afb5ceSopenharmony_ci{
156d4afb5ceSopenharmony_ci	int steps;
157d4afb5ceSopenharmony_ci
158d4afb5ceSopenharmony_ci	dest->seq = def;
159d4afb5ceSopenharmony_ci	dest->ph = def->ledphase_offset;
160d4afb5ceSopenharmony_ci	dest->phase_budget = def->ledphase_total;
161d4afb5ceSopenharmony_ci
162d4afb5ceSopenharmony_ci	/*
163d4afb5ceSopenharmony_ci	 * We need to compute the incremental phase angle step to cover the
164d4afb5ceSopenharmony_ci	 * total number of phases in the indicated ms, incrementing at the
165d4afb5ceSopenharmony_ci	 * timer rate of LWS_LED_SEQUENCER_UPDATE_RATE_HZ.  Eg,
166d4afb5ceSopenharmony_ci	 *
167d4afb5ceSopenharmony_ci	 * 65536 phase steps (one cycle) in 2000ms at 30Hz timer rate means we
168d4afb5ceSopenharmony_ci	 * will update 2000ms / 33ms = 60 times, so we must step at at
169d4afb5ceSopenharmony_ci	 * 65536 / 60 = 1092 phase angle resolution
170d4afb5ceSopenharmony_ci	 */
171d4afb5ceSopenharmony_ci
172d4afb5ceSopenharmony_ci	steps = def->ms / LWS_LED_SEQUENCER_UPDATE_INTERVAL_MS;
173d4afb5ceSopenharmony_ci	dest->step = (def->ledphase_total != LWS_SEQ_LEDPHASE_TOTAL_ENDLESS ?
174d4afb5ceSopenharmony_ci		def->ledphase_total : LWS_LED_FUNC_PHASE) / (steps ? steps : 1);
175d4afb5ceSopenharmony_ci
176d4afb5ceSopenharmony_ci	if (!lcs->timer_refcount++) {
177d4afb5ceSopenharmony_ci#if defined(LWS_PLAT_TIMER_START)
178d4afb5ceSopenharmony_ci		LWS_PLAT_TIMER_START(lcs->timer);
179d4afb5ceSopenharmony_ci#endif
180d4afb5ceSopenharmony_ci	}
181d4afb5ceSopenharmony_ci
182d4afb5ceSopenharmony_ci	return steps;
183d4afb5ceSopenharmony_ci}
184d4afb5ceSopenharmony_ci
185d4afb5ceSopenharmony_ciint
186d4afb5ceSopenharmony_cilws_led_transition(struct lws_led_state *lcs, const char *name,
187d4afb5ceSopenharmony_ci		   const lws_led_sequence_def_t *next,
188d4afb5ceSopenharmony_ci		   const lws_led_sequence_def_t *trans)
189d4afb5ceSopenharmony_ci{
190d4afb5ceSopenharmony_ci	lws_led_state_chs_t *chs = (lws_led_state_chs_t *)&lcs[1];
191d4afb5ceSopenharmony_ci	int index = lws_led_gpio_lookup(&lcs->controller->led_ops, name);
192d4afb5ceSopenharmony_ci
193d4afb5ceSopenharmony_ci	if (index < 0)
194d4afb5ceSopenharmony_ci		return 1;
195d4afb5ceSopenharmony_ci
196d4afb5ceSopenharmony_ci	lws_led_set_chs_seq(lcs, &chs[index].seqs[LLSI_TRANS], trans);
197d4afb5ceSopenharmony_ci	lws_led_set_chs_seq(lcs, &chs[index].seqs[LLSI_NEXT], next);
198d4afb5ceSopenharmony_ci
199d4afb5ceSopenharmony_ci	return 0;
200d4afb5ceSopenharmony_ci}
201