1use crate::input_device::{self, Handled, InputDeviceBinding, InputDeviceStatus, InputEvent};
6use crate::metrics;
7use anyhow::{Error, format_err};
8use async_trait::async_trait;
9use fidl_fuchsia_input_report::{
10 self as fidl_input_report, ConsumerControlButton, InputDeviceProxy, InputReport,
11};
12use fuchsia_inspect::ArrayProperty;
13use fuchsia_inspect::health::Reporter;
14
15use futures::channel::mpsc::{UnboundedReceiver, UnboundedSender};
16use metrics_registry::*;
17use zx::{AsHandleRef, HandleBased};
18
19#[derive(Debug)]
31pub struct ConsumerControlsEvent {
32 pub pressed_buttons: Vec<ConsumerControlButton>,
33 pub wake_lease: Option<zx::EventPair>,
34}
35
36impl Clone for ConsumerControlsEvent {
37 fn clone(&self) -> Self {
38 log::debug!("ConsumerControlsEvent cloned without wake lease.");
39 Self { pressed_buttons: self.pressed_buttons.clone(), wake_lease: None }
40 }
41}
42
43impl PartialEq for ConsumerControlsEvent {
44 fn eq(&self, other: &Self) -> bool {
45 self.pressed_buttons == other.pressed_buttons
46 && self.wake_lease.as_ref().map(|h| h.get_koid())
47 == other.wake_lease.as_ref().map(|h| h.get_koid())
48 }
49}
50
51impl Drop for ConsumerControlsEvent {
52 fn drop(&mut self) {
53 log::debug!("ConsumerControlsEvent dropped, had_wake_lease: {:?}", self.wake_lease);
54 }
55}
56
57impl ConsumerControlsEvent {
58 pub fn new(
63 pressed_buttons: Vec<ConsumerControlButton>,
64 wake_lease: Option<zx::EventPair>,
65 ) -> Self {
66 Self { pressed_buttons, wake_lease }
67 }
68
69 pub fn clone_with_wake_lease(&self) -> Self {
70 log::debug!("ConsumerControlsEvent cloned with wake lease: {:?}", self.wake_lease);
71 Self {
72 pressed_buttons: self.pressed_buttons.clone(),
73 wake_lease: self.wake_lease.as_ref().map(|lease| {
74 lease
75 .duplicate_handle(zx::Rights::SAME_RIGHTS)
76 .expect("failed to duplicate event pair")
77 }),
78 }
79 }
80
81 pub fn record_inspect(&self, node: &fuchsia_inspect::Node) {
82 let pressed_buttons_node =
83 node.create_string_array("pressed_buttons", self.pressed_buttons.len());
84 self.pressed_buttons.iter().enumerate().for_each(|(i, button)| {
85 let button_name: String = match button {
86 ConsumerControlButton::VolumeUp => "volume_up".into(),
87 ConsumerControlButton::VolumeDown => "volume_down".into(),
88 ConsumerControlButton::Pause => "pause".into(),
89 ConsumerControlButton::FactoryReset => "factory_reset".into(),
90 ConsumerControlButton::MicMute => "mic_mute".into(),
91 ConsumerControlButton::Reboot => "reboot".into(),
92 ConsumerControlButton::CameraDisable => "camera_disable".into(),
93 ConsumerControlButton::Power => "power".into(),
94 ConsumerControlButton::Function => "function".into(),
95 unknown_value => {
96 format!("unknown({:?})", unknown_value)
97 }
98 };
99 pressed_buttons_node.set(i, &button_name);
100 });
101 node.record(pressed_buttons_node);
102 }
103}
104
105pub struct ConsumerControlsBinding {
112 event_sender: UnboundedSender<input_device::InputEvent>,
114
115 device_descriptor: ConsumerControlsDeviceDescriptor,
117}
118
119#[derive(Clone, Debug, Eq, PartialEq)]
120pub struct ConsumerControlsDeviceDescriptor {
121 pub buttons: Vec<ConsumerControlButton>,
123 pub device_id: u32,
125}
126
127#[async_trait]
128impl input_device::InputDeviceBinding for ConsumerControlsBinding {
129 fn input_event_sender(&self) -> UnboundedSender<input_device::InputEvent> {
130 self.event_sender.clone()
131 }
132
133 fn get_device_descriptor(&self) -> input_device::InputDeviceDescriptor {
134 input_device::InputDeviceDescriptor::ConsumerControls(self.device_descriptor.clone())
135 }
136}
137
138impl ConsumerControlsBinding {
139 pub async fn new(
154 device_proxy: InputDeviceProxy,
155 device_id: u32,
156 input_event_sender: UnboundedSender<input_device::InputEvent>,
157 device_node: fuchsia_inspect::Node,
158 metrics_logger: metrics::MetricsLogger,
159 ) -> Result<Self, Error> {
160 let (device_binding, mut inspect_status) =
161 Self::bind_device(&device_proxy, device_id, input_event_sender, device_node).await?;
162 inspect_status.health_node.set_ok();
163 input_device::initialize_report_stream(
164 device_proxy,
165 device_binding.get_device_descriptor(),
166 device_binding.input_event_sender(),
167 inspect_status,
168 metrics_logger,
169 Self::process_reports,
170 );
171
172 Ok(device_binding)
173 }
174
175 async fn bind_device(
187 device: &InputDeviceProxy,
188 device_id: u32,
189 input_event_sender: UnboundedSender<input_device::InputEvent>,
190 device_node: fuchsia_inspect::Node,
191 ) -> Result<(Self, InputDeviceStatus), Error> {
192 let mut input_device_status = InputDeviceStatus::new(device_node);
193 let device_descriptor: fidl_input_report::DeviceDescriptor = match device
194 .get_descriptor()
195 .await
196 {
197 Ok(descriptor) => descriptor,
198 Err(_) => {
199 input_device_status.health_node.set_unhealthy("Could not get device descriptor.");
200 return Err(format_err!("Could not get descriptor for device_id: {}", device_id));
201 }
202 };
203
204 let consumer_controls_descriptor = device_descriptor.consumer_control.ok_or_else(|| {
205 input_device_status
206 .health_node
207 .set_unhealthy("DeviceDescriptor does not have a ConsumerControlDescriptor.");
208 format_err!("DeviceDescriptor does not have a ConsumerControlDescriptor")
209 })?;
210
211 let consumer_controls_input_descriptor =
212 consumer_controls_descriptor.input.ok_or_else(|| {
213 input_device_status.health_node.set_unhealthy(
214 "ConsumerControlDescriptor does not have a ConsumerControlInputDescriptor.",
215 );
216 format_err!(
217 "ConsumerControlDescriptor does not have a ConsumerControlInputDescriptor"
218 )
219 })?;
220
221 let device_descriptor: ConsumerControlsDeviceDescriptor =
222 ConsumerControlsDeviceDescriptor {
223 buttons: consumer_controls_input_descriptor.buttons.unwrap_or_default(),
224 device_id,
225 };
226
227 Ok((
228 ConsumerControlsBinding { event_sender: input_event_sender, device_descriptor },
229 input_device_status,
230 ))
231 }
232
233 fn process_reports(
254 mut report: InputReport,
255 previous_report: Option<InputReport>,
256 device_descriptor: &input_device::InputDeviceDescriptor,
257 input_event_sender: &mut UnboundedSender<input_device::InputEvent>,
258 inspect_status: &InputDeviceStatus,
259 metrics_logger: &metrics::MetricsLogger,
260 ) -> (Option<InputReport>, Option<UnboundedReceiver<InputEvent>>) {
261 fuchsia_trace::duration!(c"input", c"consumer-controls-binding-process-report");
262 if let Some(trace_id) = report.trace_id {
263 fuchsia_trace::flow_end!(c"input", c"input_report", trace_id.into());
264 }
265
266 inspect_status.count_received_report(&report);
267 let pressed_buttons: Vec<ConsumerControlButton> = match report.consumer_control {
269 Some(ref consumer_control_report) => consumer_control_report
270 .pressed_buttons
271 .as_ref()
272 .map(|buttons| buttons.iter().cloned().collect())
273 .unwrap_or_default(),
274 None => {
275 inspect_status.count_filtered_report();
276 return (previous_report, None);
277 }
278 };
279
280 let wake_lease = report.wake_lease.take();
281 let trace_id = fuchsia_trace::Id::random();
282 fuchsia_trace::flow_begin!(c"input", c"event_in_input_pipeline", trace_id);
283
284 send_consumer_controls_event(
285 pressed_buttons,
286 wake_lease,
287 device_descriptor,
288 input_event_sender,
289 inspect_status,
290 metrics_logger,
291 trace_id,
292 );
293
294 (Some(report), None)
295 }
296}
297
298fn send_consumer_controls_event(
308 pressed_buttons: Vec<ConsumerControlButton>,
309 wake_lease: Option<zx::EventPair>,
310 device_descriptor: &input_device::InputDeviceDescriptor,
311 sender: &mut UnboundedSender<input_device::InputEvent>,
312 inspect_status: &InputDeviceStatus,
313 metrics_logger: &metrics::MetricsLogger,
314 trace_id: fuchsia_trace::Id,
315) {
316 let event = input_device::InputEvent {
317 device_event: input_device::InputDeviceEvent::ConsumerControls(ConsumerControlsEvent::new(
318 pressed_buttons,
319 wake_lease,
320 )),
321 device_descriptor: device_descriptor.clone(),
322 event_time: zx::MonotonicInstant::get(),
323 handled: Handled::No,
324 trace_id: Some(trace_id),
325 };
326
327 match sender.unbounded_send(event.clone_with_wake_lease()) {
328 Err(e) => metrics_logger.log_error(
329 InputPipelineErrorMetricDimensionEvent::ConsumerControlsSendEventFailed,
330 std::format!("Failed to send ConsumerControlsEvent with error: {:?}", e),
331 ),
332 _ => inspect_status.count_generated_event(event),
333 }
334}
335
336#[cfg(test)]
337mod tests {
338 use super::*;
339 use crate::testing_utilities;
340 use fuchsia_async as fasync;
341 use futures::StreamExt;
342
343 #[fasync::run_singlethreaded(test)]
346 async fn volume_up_only() {
347 let (event_time_i64, event_time_u64) = testing_utilities::event_times();
348 let pressed_buttons = vec![ConsumerControlButton::VolumeUp];
349 let first_report = testing_utilities::create_consumer_control_input_report(
350 pressed_buttons.clone(),
351 event_time_i64,
352 );
353 let descriptor = testing_utilities::consumer_controls_device_descriptor();
354
355 let input_reports = vec![first_report];
356 let expected_events = vec![testing_utilities::create_consumer_controls_event(
357 pressed_buttons,
358 event_time_u64,
359 &descriptor,
360 )];
361
362 assert_input_report_sequence_generates_events!(
363 input_reports: input_reports,
364 expected_events: expected_events,
365 device_descriptor: descriptor,
366 device_type: ConsumerControlsBinding,
367 );
368 }
369
370 #[fasync::run_singlethreaded(test)]
373 async fn volume_up_and_down() {
374 let (event_time_i64, event_time_u64) = testing_utilities::event_times();
375 let pressed_buttons =
376 vec![ConsumerControlButton::VolumeUp, ConsumerControlButton::VolumeDown];
377 let first_report = testing_utilities::create_consumer_control_input_report(
378 pressed_buttons.clone(),
379 event_time_i64,
380 );
381 let descriptor = testing_utilities::consumer_controls_device_descriptor();
382
383 let input_reports = vec![first_report];
384 let expected_events = vec![testing_utilities::create_consumer_controls_event(
385 pressed_buttons,
386 event_time_u64,
387 &descriptor,
388 )];
389
390 assert_input_report_sequence_generates_events!(
391 input_reports: input_reports,
392 expected_events: expected_events,
393 device_descriptor: descriptor,
394 device_type: ConsumerControlsBinding,
395 );
396 }
397
398 #[fasync::run_singlethreaded(test)]
401 async fn sequence_of_buttons() {
402 let (event_time_i64, event_time_u64) = testing_utilities::event_times();
403 let first_report = testing_utilities::create_consumer_control_input_report(
404 vec![ConsumerControlButton::VolumeUp],
405 event_time_i64,
406 );
407 let second_report = testing_utilities::create_consumer_control_input_report(
408 vec![ConsumerControlButton::VolumeDown],
409 event_time_i64,
410 );
411 let third_report = testing_utilities::create_consumer_control_input_report(
412 vec![ConsumerControlButton::CameraDisable],
413 event_time_i64,
414 );
415 let descriptor = testing_utilities::consumer_controls_device_descriptor();
416
417 let input_reports = vec![first_report, second_report, third_report];
418 let expected_events = vec![
419 testing_utilities::create_consumer_controls_event(
420 vec![ConsumerControlButton::VolumeUp],
421 event_time_u64,
422 &descriptor,
423 ),
424 testing_utilities::create_consumer_controls_event(
425 vec![ConsumerControlButton::VolumeDown],
426 event_time_u64,
427 &descriptor,
428 ),
429 testing_utilities::create_consumer_controls_event(
430 vec![ConsumerControlButton::CameraDisable],
431 event_time_u64,
432 &descriptor,
433 ),
434 ];
435
436 assert_input_report_sequence_generates_events!(
437 input_reports: input_reports,
438 expected_events: expected_events,
439 device_descriptor: descriptor,
440 device_type: ConsumerControlsBinding,
441 );
442 }
443}