blob: 2760334a36aa38895f946f0cc4c34c783f915ce8 [file] [log] [blame]
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -04001/*
Ramakant Singhd9b17902017-05-16 12:49:56 +05302* Copyright (c) 2014, 2016-2017, The Linux Foundation. All rights reserved.
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -04003*
4* Redistribution and use in source and binary forms, with or without
5* modification, are permitted provided that the following conditions are
6* met:
7* * Redistributions of source code must retain the above copyright
8* notice, this list of conditions and the following disclaimer.
9* * Redistributions in binary form must reproduce the above
10* copyright notice, this list of conditions and the following
11* disclaimer in the documentation and/or other materials provided
12* with the distribution.
13* * Neither the name of The Linux Foundation. nor the names of its
14* contributors may be used to endorse or promote products derived
15* from this software without specific prior written permission.
16*
17* THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27* IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*/
29
Naseer Ahmed44e8a9a2015-02-24 15:33:23 -050030#define DEBUG 0
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -040031#define ATRACE_TAG (ATRACE_TAG_GRAPHICS | ATRACE_TAG_HAL)
32#include <cstdlib>
Naseer Ahmedfcad05e2018-03-06 20:41:14 -050033#include <log/log.h>
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -040034#include <errno.h>
35#include <fcntl.h>
36#include <hardware/hdmi_cec.h>
37#include <utils/Trace.h>
38#include "qhdmi_cec.h"
39#include "QHDMIClient.h"
40
41namespace qhdmicec {
42
43const int NUM_HDMI_PORTS = 1;
44const int MAX_SYSFS_DATA = 128;
45const int MAX_CEC_FRAME_SIZE = 20;
46const int MAX_SEND_MESSAGE_RETRIES = 1;
47
48enum {
49 LOGICAL_ADDRESS_SET = 1,
50 LOGICAL_ADDRESS_UNSET = -1,
51};
52
53// Offsets of members of struct hdmi_cec_msg
54// drivers/video/msm/mdss/mdss_hdmi_cec.c
55// XXX: Get this from a driver header
56enum {
57 CEC_OFFSET_SENDER_ID,
58 CEC_OFFSET_RECEIVER_ID,
59 CEC_OFFSET_OPCODE,
60 CEC_OFFSET_OPERAND,
Tatenda Chipeperekwa13378a02016-04-01 12:21:24 -070061 CEC_OFFSET_FRAME_LENGTH = 17,
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -040062 CEC_OFFSET_RETRANSMIT,
63};
64
65//Forward declarations
66static void cec_close_context(cec_context_t* ctx __unused);
67static int cec_enable(cec_context_t *ctx, int enable);
Naseer Ahmed44e8a9a2015-02-24 15:33:23 -050068static int cec_is_connected(const struct hdmi_cec_device* dev, int port_id);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -040069
70static ssize_t read_node(const char *path, char *data)
71{
72 ssize_t err = 0;
73 FILE *fp = NULL;
74 err = access(path, R_OK);
75 if (!err) {
76 fp = fopen(path, "r");
77 if (fp) {
78 err = fread(data, sizeof(char), MAX_SYSFS_DATA ,fp);
79 fclose(fp);
80 }
81 }
82 return err;
83}
84
85static ssize_t write_node(const char *path, const char *data, size_t len)
86{
87 ssize_t err = 0;
88 int fd = -1;
89 err = access(path, W_OK);
90 if (!err) {
91 fd = open(path, O_WRONLY);
92 errno = 0;
93 err = write(fd, data, len);
94 if (err < 0) {
95 err = -errno;
96 }
97 close(fd);
98 } else {
99 ALOGE("%s: Failed to access path: %s error: %s",
100 __FUNCTION__, path, strerror(errno));
101 err = -errno;
102 }
103 return err;
104}
105
106// Helper function to write integer values to the full sysfs path
107static ssize_t write_int_to_node(cec_context_t *ctx,
108 const char *path_postfix,
109 const int value)
110{
111 char sysfs_full_path[MAX_PATH_LENGTH];
112 char sysfs_data[MAX_SYSFS_DATA];
113 snprintf(sysfs_data, sizeof(sysfs_data), "%d",value);
114 snprintf(sysfs_full_path,sizeof(sysfs_full_path), "%s/%s",
115 ctx->fb_sysfs_path, path_postfix);
116 ssize_t err = write_node(sysfs_full_path, sysfs_data, strlen(sysfs_data));
117 return err;
118}
119
120static void hex_to_string(const char *msg, ssize_t len, char *str)
121{
122 //Functions assumes sufficient memory in str
123 char *ptr = str;
124 for(int i=0; i < len ; i++) {
125 ptr += snprintf(ptr, 3, "%02X", msg[i]);
126 // Overwrite null termination of snprintf in all except the last byte
127 if (i < len - 1)
128 *ptr = ':';
129 ptr++;
130 }
131}
132
133static ssize_t cec_get_fb_node_number(cec_context_t *ctx)
134{
135 //XXX: Do this from a common utility library across the display HALs
136 const int MAX_FB_DEVICES = 2;
137 ssize_t len = 0;
138 char fb_type_path[MAX_PATH_LENGTH];
139 char fb_type[MAX_SYSFS_DATA];
140 const char *dtv_panel_str = "dtv panel";
141
142 for(int num = 0; num < MAX_FB_DEVICES; num++) {
143 snprintf(fb_type_path, sizeof(fb_type_path),"%s%d/msm_fb_type",
144 SYSFS_BASE,num);
145 ALOGD_IF(DEBUG, "%s: num: %d fb_type_path: %s", __FUNCTION__, num, fb_type_path);
146 len = read_node(fb_type_path, fb_type);
147 ALOGD_IF(DEBUG, "%s: fb_type:%s", __FUNCTION__, fb_type);
148 if(len > 0 && (strncmp(fb_type, dtv_panel_str, strlen(dtv_panel_str)) == 0)){
149 ALOGD_IF(DEBUG, "%s: Found DTV panel at fb%d", __FUNCTION__, num);
150 ctx->fb_num = num;
151 snprintf(ctx->fb_sysfs_path, sizeof(ctx->fb_sysfs_path),
152 "%s%d", SYSFS_BASE, num);
153 break;
154 }
155 }
156 if (len < 0)
157 return len;
158 else
159 return 0;
160}
161
162static int cec_add_logical_address(const struct hdmi_cec_device* dev,
163 cec_logical_address_t addr)
164{
165 if (addr < CEC_ADDR_TV || addr > CEC_ADDR_BROADCAST) {
166 ALOGE("%s: Received invalid address: %d ", __FUNCTION__, addr);
167 return -EINVAL;
168 }
169 cec_context_t* ctx = (cec_context_t*)(dev);
170 ctx->logical_address[addr] = LOGICAL_ADDRESS_SET;
171
172 //XXX: We can get multiple logical addresses here but we can only send one
173 //to the driver. Store locally for now
174 ssize_t err = write_int_to_node(ctx, "cec/logical_addr", addr);
Naseer Ahmed44e8a9a2015-02-24 15:33:23 -0500175 ALOGI("%s: Allocated logical address: %d ", __FUNCTION__, addr);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400176 return (int) err;
177}
178
179static void cec_clear_logical_address(const struct hdmi_cec_device* dev)
180{
181 cec_context_t* ctx = (cec_context_t*)(dev);
182 memset(ctx->logical_address, LOGICAL_ADDRESS_UNSET,
183 sizeof(ctx->logical_address));
184 //XXX: Find logical_addr that needs to be reset
Naseer Ahmedf51812c2014-10-28 15:37:18 -0400185 write_int_to_node(ctx, "cec/logical_addr", 15);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400186 ALOGD_IF(DEBUG, "%s: Cleared logical addresses", __FUNCTION__);
187}
188
189static int cec_get_physical_address(const struct hdmi_cec_device* dev,
190 uint16_t* addr)
191{
192 cec_context_t* ctx = (cec_context_t*)(dev);
Naseer Ahmed251c0302014-10-10 12:53:13 -0400193 char pa_path[MAX_PATH_LENGTH];
194 char pa_data[MAX_SYSFS_DATA];
195 snprintf (pa_path, sizeof(pa_path),"%s/pa",
196 ctx->fb_sysfs_path);
197 int err = (int) read_node(pa_path, pa_data);
198 *addr = (uint16_t) atoi(pa_data);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400199 ALOGD_IF(DEBUG, "%s: Physical Address: 0x%x", __FUNCTION__, *addr);
Naseer Ahmed251c0302014-10-10 12:53:13 -0400200 if (err < 0)
201 return err;
202 else
203 return 0;
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400204}
205
206static int cec_send_message(const struct hdmi_cec_device* dev,
207 const cec_message_t* msg)
208{
209 ATRACE_CALL();
Naseer Ahmed44e8a9a2015-02-24 15:33:23 -0500210 if(cec_is_connected(dev, 0) <= 0)
211 return HDMI_RESULT_FAIL;
212
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400213 cec_context_t* ctx = (cec_context_t*)(dev);
214 ALOGD_IF(DEBUG, "%s: initiator: %d destination: %d length: %u",
215 __FUNCTION__, msg->initiator, msg->destination,
216 (uint32_t) msg->length);
217
218 // Dump message received from framework
219 char dump[128];
220 if(msg->length > 0) {
221 hex_to_string((char*)msg->body, msg->length, dump);
222 ALOGD_IF(DEBUG, "%s: message from framework: %s", __FUNCTION__, dump);
223 }
224
225 char write_msg_path[MAX_PATH_LENGTH];
226 char write_msg[MAX_CEC_FRAME_SIZE];
227 memset(write_msg, 0, sizeof(write_msg));
228 // See definition of struct hdmi_cec_msg in driver code
229 // drivers/video/msm/mdss/mdss_hdmi_cec.c
230 // Write header block
231 // XXX: Include this from header in kernel
232 write_msg[CEC_OFFSET_SENDER_ID] = msg->initiator;
233 write_msg[CEC_OFFSET_RECEIVER_ID] = msg->destination;
234 //Kernel splits opcode/operand, but Android sends it in one byte array
235 write_msg[CEC_OFFSET_OPCODE] = msg->body[0];
236 if(msg->length > 1) {
237 memcpy(&write_msg[CEC_OFFSET_OPERAND], &msg->body[1],
238 sizeof(char)*(msg->length - 1));
239 }
240 //msg length + initiator + destination
241 write_msg[CEC_OFFSET_FRAME_LENGTH] = (unsigned char) (msg->length + 1);
242 hex_to_string(write_msg, sizeof(write_msg), dump);
243 ALOGD_IF(DEBUG, "%s: message to driver: %s", __FUNCTION__, dump);
244 snprintf(write_msg_path, sizeof(write_msg_path), "%s/cec/wr_msg",
245 ctx->fb_sysfs_path);
246 int retry_count = 0;
247 ssize_t err = 0;
248 //HAL spec requires us to retry at least once.
249 while (true) {
250 err = write_node(write_msg_path, write_msg, sizeof(write_msg));
251 retry_count++;
252 if (err == -EAGAIN && retry_count <= MAX_SEND_MESSAGE_RETRIES) {
253 ALOGE("%s: CEC line busy, retrying", __FUNCTION__);
254 } else {
255 break;
256 }
257 }
258
259 if (err < 0) {
260 if (err == -ENXIO) {
Naseer Ahmed44e8a9a2015-02-24 15:33:23 -0500261 ALOGI("%s: No device exists with the destination address",
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400262 __FUNCTION__);
263 return HDMI_RESULT_NACK;
264 } else if (err == -EAGAIN) {
265 ALOGE("%s: CEC line is busy, max retry count exceeded",
266 __FUNCTION__);
267 return HDMI_RESULT_BUSY;
268 } else {
269 return HDMI_RESULT_FAIL;
270 ALOGE("%s: Failed to send CEC message err: %zd - %s",
271 __FUNCTION__, err, strerror(int(-err)));
272 }
273 } else {
274 ALOGD_IF(DEBUG, "%s: Sent CEC message - %zd bytes written",
275 __FUNCTION__, err);
276 return HDMI_RESULT_SUCCESS;
277 }
278}
279
280void cec_receive_message(cec_context_t *ctx, char *msg, ssize_t len)
281{
Naseer Ahmed251c0302014-10-10 12:53:13 -0400282 if(!ctx->system_control)
283 return;
284
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400285 char dump[128];
286 if(len > 0) {
287 hex_to_string(msg, len, dump);
288 ALOGD_IF(DEBUG, "%s: Message from driver: %s", __FUNCTION__, dump);
289 }
290
291 hdmi_event_t event;
292 event.type = HDMI_EVENT_CEC_MESSAGE;
293 event.dev = (hdmi_cec_device *) ctx;
294 // Remove initiator/destination from this calculation
295 event.cec.length = msg[CEC_OFFSET_FRAME_LENGTH] - 1;
296 event.cec.initiator = (cec_logical_address_t) msg[CEC_OFFSET_SENDER_ID];
297 event.cec.destination = (cec_logical_address_t) msg[CEC_OFFSET_RECEIVER_ID];
298 //Copy opcode and operand
Ramakant Singhd9b17902017-05-16 12:49:56 +0530299 size_t copy_size = event.cec.length > sizeof(event.cec.body) ?
300 sizeof(event.cec.body) : event.cec.length;
301 memcpy(event.cec.body, &msg[CEC_OFFSET_OPCODE],copy_size);
302 hex_to_string((char *) event.cec.body, copy_size, dump);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400303 ALOGD_IF(DEBUG, "%s: Message to framework: %s", __FUNCTION__, dump);
304 ctx->callback.callback_func(&event, ctx->callback.callback_arg);
305}
306
307void cec_hdmi_hotplug(cec_context_t *ctx, int connected)
308{
Naseer Ahmed251c0302014-10-10 12:53:13 -0400309 //Ignore unplug events when system control is disabled
310 if(!ctx->system_control && connected == 0)
311 return;
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400312 hdmi_event_t event;
313 event.type = HDMI_EVENT_HOT_PLUG;
314 event.dev = (hdmi_cec_device *) ctx;
315 event.hotplug.connected = connected ? HDMI_CONNECTED : HDMI_NOT_CONNECTED;
316 ctx->callback.callback_func(&event, ctx->callback.callback_arg);
317}
318
319static void cec_register_event_callback(const struct hdmi_cec_device* dev,
320 event_callback_t callback, void* arg)
321{
322 ALOGD_IF(DEBUG, "%s: Registering callback", __FUNCTION__);
323 cec_context_t* ctx = (cec_context_t*)(dev);
324 ctx->callback.callback_func = callback;
325 ctx->callback.callback_arg = arg;
326}
327
328static void cec_get_version(const struct hdmi_cec_device* dev, int* version)
329{
330 cec_context_t* ctx = (cec_context_t*)(dev);
331 *version = ctx->version;
332 ALOGD_IF(DEBUG, "%s: version: %d", __FUNCTION__, *version);
333}
334
335static void cec_get_vendor_id(const struct hdmi_cec_device* dev,
336 uint32_t* vendor_id)
337{
338 cec_context_t* ctx = (cec_context_t*)(dev);
339 *vendor_id = ctx->vendor_id;
340 ALOGD_IF(DEBUG, "%s: vendor id: %u", __FUNCTION__, *vendor_id);
341}
342
343static void cec_get_port_info(const struct hdmi_cec_device* dev,
344 struct hdmi_port_info* list[], int* total)
345{
346 ALOGD_IF(DEBUG, "%s: Get port info", __FUNCTION__);
347 cec_context_t* ctx = (cec_context_t*)(dev);
348 *total = NUM_HDMI_PORTS;
349 *list = ctx->port_info;
350}
351
352static void cec_set_option(const struct hdmi_cec_device* dev, int flag,
353 int value)
354{
355 cec_context_t* ctx = (cec_context_t*)(dev);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400356 switch (flag) {
357 case HDMI_OPTION_WAKEUP:
Naseer Ahmed251c0302014-10-10 12:53:13 -0400358 ALOGD_IF(DEBUG, "%s: Wakeup: value: %d", __FUNCTION__, value);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400359 //XXX
360 break;
361 case HDMI_OPTION_ENABLE_CEC:
Naseer Ahmed251c0302014-10-10 12:53:13 -0400362 ALOGD_IF(DEBUG, "%s: Enable CEC: value: %d", __FUNCTION__, value);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400363 cec_enable(ctx, value? 1 : 0);
364 break;
365 case HDMI_OPTION_SYSTEM_CEC_CONTROL:
Naseer Ahmed251c0302014-10-10 12:53:13 -0400366 ALOGD_IF(DEBUG, "%s: system_control: value: %d",
367 __FUNCTION__, value);
368 ctx->system_control = !!value;
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400369 break;
370 }
371}
372
373static void cec_set_audio_return_channel(const struct hdmi_cec_device* dev,
Naseer Ahmed6f4eaf02015-03-20 17:59:25 -0400374 int port, int flag)
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400375{
376 cec_context_t* ctx = (cec_context_t*)(dev);
377 ctx->arc_enabled = flag ? true : false;
Naseer Ahmed6f4eaf02015-03-20 17:59:25 -0400378 ALOGD_IF(DEBUG, "%s: ARC flag: %d port: %d", __FUNCTION__, flag, port);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400379}
380
381static int cec_is_connected(const struct hdmi_cec_device* dev, int port_id)
382{
383 // Ignore port_id since we have only one port
384 int connected = 0;
385 cec_context_t* ctx = (cec_context_t*)(dev);
386 char connected_path[MAX_PATH_LENGTH];
387 char connected_data[MAX_SYSFS_DATA];
388 snprintf (connected_path, sizeof(connected_path),"%s/connected",
389 ctx->fb_sysfs_path);
390 ssize_t err = read_node(connected_path, connected_data);
391 connected = atoi(connected_data);
392
393 ALOGD_IF(DEBUG, "%s: HDMI at port %d is - %s", __FUNCTION__, port_id,
394 connected ? "connected":"disconnected");
Naseer Ahmed251c0302014-10-10 12:53:13 -0400395 if (err < 0)
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400396 return (int) err;
397 else
398 return connected;
399}
400
401static int cec_device_close(struct hw_device_t *dev)
402{
403 ALOGD_IF(DEBUG, "%s: Close CEC HAL ", __FUNCTION__);
404 if (!dev) {
405 ALOGE("%s: NULL device pointer", __FUNCTION__);
406 return -EINVAL;
407 }
408 cec_context_t* ctx = (cec_context_t*)(dev);
409 cec_close_context(ctx);
410 free(dev);
411 return 0;
412}
413
414static int cec_enable(cec_context_t *ctx, int enable)
415{
416 ssize_t err;
Naseer Ahmed251c0302014-10-10 12:53:13 -0400417 // Enable CEC
Tatenda Chipeperekwa13378a02016-04-01 12:21:24 -0700418 int value = enable ? 0x3 : 0x0;
Naseer Ahmed251c0302014-10-10 12:53:13 -0400419 err = write_int_to_node(ctx, "cec/enable", value);
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400420 if(err < 0) {
421 ALOGE("%s: Failed to toggle CEC: enable: %d",
422 __FUNCTION__, enable);
423 return (int) err;
424 }
425 ctx->enabled = enable;
426 return 0;
427}
428
429static void cec_init_context(cec_context_t *ctx)
430{
431 ALOGD_IF(DEBUG, "%s: Initializing context", __FUNCTION__);
432 cec_get_fb_node_number(ctx);
433
434 //Initialize ports - We support only one output port
435 ctx->port_info = new hdmi_port_info[NUM_HDMI_PORTS];
436 ctx->port_info[0].type = HDMI_OUTPUT;
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400437 ctx->port_info[0].port_id = 1;
438 ctx->port_info[0].cec_supported = 1;
439 //XXX: Enable ARC if supported
440 ctx->port_info[0].arc_supported = 0;
Naseer Ahmed251c0302014-10-10 12:53:13 -0400441 cec_get_physical_address((hdmi_cec_device *) ctx,
442 &ctx->port_info[0].physical_address );
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400443
444 ctx->version = 0x4;
Naseer Ahmed251c0302014-10-10 12:53:13 -0400445 ctx->vendor_id = 0xA47733;
Naseer Ahmed7a7b66d2014-07-23 17:56:26 -0400446 cec_clear_logical_address((hdmi_cec_device_t*)ctx);
447
448 //Set up listener for HDMI events
449 ctx->disp_client = new qClient::QHDMIClient();
450 ctx->disp_client->setCECContext(ctx);
451 ctx->disp_client->registerClient(ctx->disp_client);
452
453 //Enable CEC - framework expects it to be enabled by default
454 cec_enable(ctx, true);
455
456 ALOGD("%s: CEC enabled", __FUNCTION__);
457}
458
459static void cec_close_context(cec_context_t* ctx __unused)
460{
461 ALOGD("%s: Closing context", __FUNCTION__);
462}
463
464static int cec_device_open(const struct hw_module_t* module,
465 const char* name,
466 struct hw_device_t** device)
467{
468 ALOGD_IF(DEBUG, "%s: name: %s", __FUNCTION__, name);
469 int status = -EINVAL;
470 if (!strcmp(name, HDMI_CEC_HARDWARE_INTERFACE )) {
471 struct cec_context_t *dev;
472 dev = (cec_context_t *) calloc (1, sizeof(*dev));
473 if (dev) {
474 cec_init_context(dev);
475
476 //Setup CEC methods
477 dev->device.common.tag = HARDWARE_DEVICE_TAG;
478 dev->device.common.version = HDMI_CEC_DEVICE_API_VERSION_1_0;
479 dev->device.common.module = const_cast<hw_module_t* >(module);
480 dev->device.common.close = cec_device_close;
481 dev->device.add_logical_address = cec_add_logical_address;
482 dev->device.clear_logical_address = cec_clear_logical_address;
483 dev->device.get_physical_address = cec_get_physical_address;
484 dev->device.send_message = cec_send_message;
485 dev->device.register_event_callback = cec_register_event_callback;
486 dev->device.get_version = cec_get_version;
487 dev->device.get_vendor_id = cec_get_vendor_id;
488 dev->device.get_port_info = cec_get_port_info;
489 dev->device.set_option = cec_set_option;
490 dev->device.set_audio_return_channel = cec_set_audio_return_channel;
491 dev->device.is_connected = cec_is_connected;
492
493 *device = &dev->device.common;
494 status = 0;
495 } else {
496 status = -EINVAL;
497 }
498 }
499 return status;
500}
501}; //namespace qhdmicec
502
503// Standard HAL module, should be outside qhdmicec namespace
504static struct hw_module_methods_t cec_module_methods = {
505 .open = qhdmicec::cec_device_open
506};
507
508hdmi_module_t HAL_MODULE_INFO_SYM = {
509 .common = {
510 .tag = HARDWARE_MODULE_TAG,
511 .version_major = 1,
512 .version_minor = 0,
513 .id = HDMI_CEC_HARDWARE_MODULE_ID,
514 .name = "QTI HDMI CEC module",
515 .author = "The Linux Foundation",
516 .methods = &cec_module_methods,
517 }
518};
519
520