mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-15 09:34:17 +00:00
6ec1cbf6b1
It turns out that the struct cec_fh event buffer size of 64 events (64 for CEC_EVENT_PIN_CEC_LOW and 64 for _HIGH) is too small. It's about 160 ms worth of events and if the Raspberry Pi is busy, then it might take too long for the application to be scheduled so that it can drain the pending events. Increase these buffers to 800 events which is at least 2 seconds worth of events. There is also a FIFO in between the interrupt and the cec-pin thread. The thread passes the events on to the CEC core. It is important that should this FIFO fill up the cec core will be informed that events have been lost so this can be communicated to the user by setting CEC_EVENT_FL_DROPPED_EVENTS. It is very hard to debug CEC problems if events were lost without informing the user of that fact. If events were dropped due to the FIFO filling up, then the debugfs status file will let you know how many events were dropped. Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
288 lines
7.8 KiB
C
288 lines
7.8 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* vivid-cec.c - A Virtual Video Test Driver, cec emulation
|
|
*
|
|
* Copyright 2016 Cisco Systems, Inc. and/or its affiliates. All rights reserved.
|
|
*/
|
|
|
|
#include <media/cec.h>
|
|
|
|
#include "vivid-core.h"
|
|
#include "vivid-cec.h"
|
|
|
|
#define CEC_TIM_START_BIT_TOTAL 4500
|
|
#define CEC_TIM_START_BIT_LOW 3700
|
|
#define CEC_TIM_START_BIT_HIGH 800
|
|
#define CEC_TIM_DATA_BIT_TOTAL 2400
|
|
#define CEC_TIM_DATA_BIT_0_LOW 1500
|
|
#define CEC_TIM_DATA_BIT_0_HIGH 900
|
|
#define CEC_TIM_DATA_BIT_1_LOW 600
|
|
#define CEC_TIM_DATA_BIT_1_HIGH 1800
|
|
|
|
void vivid_cec_bus_free_work(struct vivid_dev *dev)
|
|
{
|
|
spin_lock(&dev->cec_slock);
|
|
while (!list_empty(&dev->cec_work_list)) {
|
|
struct vivid_cec_work *cw =
|
|
list_first_entry(&dev->cec_work_list,
|
|
struct vivid_cec_work, list);
|
|
|
|
spin_unlock(&dev->cec_slock);
|
|
cancel_delayed_work_sync(&cw->work);
|
|
spin_lock(&dev->cec_slock);
|
|
list_del(&cw->list);
|
|
cec_transmit_attempt_done(cw->adap, CEC_TX_STATUS_LOW_DRIVE);
|
|
kfree(cw);
|
|
}
|
|
spin_unlock(&dev->cec_slock);
|
|
}
|
|
|
|
static bool vivid_cec_find_dest_adap(struct vivid_dev *dev,
|
|
struct cec_adapter *adap, u8 dest)
|
|
{
|
|
unsigned int i;
|
|
|
|
if (dest >= 0xf)
|
|
return false;
|
|
|
|
if (adap != dev->cec_rx_adap && dev->cec_rx_adap &&
|
|
dev->cec_rx_adap->is_configured &&
|
|
cec_has_log_addr(dev->cec_rx_adap, dest))
|
|
return true;
|
|
|
|
for (i = 0; i < MAX_OUTPUTS && dev->cec_tx_adap[i]; i++) {
|
|
if (adap == dev->cec_tx_adap[i])
|
|
continue;
|
|
if (!dev->cec_tx_adap[i]->is_configured)
|
|
continue;
|
|
if (cec_has_log_addr(dev->cec_tx_adap[i], dest))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static void vivid_cec_pin_adap_events(struct cec_adapter *adap, ktime_t ts,
|
|
const struct cec_msg *msg, bool nacked)
|
|
{
|
|
unsigned int len = nacked ? 1 : msg->len;
|
|
unsigned int i;
|
|
bool bit;
|
|
|
|
if (adap == NULL)
|
|
return;
|
|
|
|
/*
|
|
* Suffix ULL on constant 10 makes the expression
|
|
* CEC_TIM_START_BIT_TOTAL + 10ULL * len * CEC_TIM_DATA_BIT_TOTAL
|
|
* to be evaluated using 64-bit unsigned arithmetic (u64), which
|
|
* is what ktime_sub_us expects as second argument.
|
|
*/
|
|
ts = ktime_sub_us(ts, CEC_TIM_START_BIT_TOTAL +
|
|
10ULL * len * CEC_TIM_DATA_BIT_TOTAL);
|
|
cec_queue_pin_cec_event(adap, false, false, ts);
|
|
ts = ktime_add_us(ts, CEC_TIM_START_BIT_LOW);
|
|
cec_queue_pin_cec_event(adap, true, false, ts);
|
|
ts = ktime_add_us(ts, CEC_TIM_START_BIT_HIGH);
|
|
|
|
for (i = 0; i < 10 * len; i++) {
|
|
switch (i % 10) {
|
|
case 0 ... 7:
|
|
bit = msg->msg[i / 10] & (0x80 >> (i % 10));
|
|
break;
|
|
case 8: /* EOM */
|
|
bit = i / 10 == msg->len - 1;
|
|
break;
|
|
case 9: /* ACK */
|
|
bit = cec_msg_is_broadcast(msg) ^ nacked;
|
|
break;
|
|
}
|
|
cec_queue_pin_cec_event(adap, false, false, ts);
|
|
if (bit)
|
|
ts = ktime_add_us(ts, CEC_TIM_DATA_BIT_1_LOW);
|
|
else
|
|
ts = ktime_add_us(ts, CEC_TIM_DATA_BIT_0_LOW);
|
|
cec_queue_pin_cec_event(adap, true, false, ts);
|
|
if (bit)
|
|
ts = ktime_add_us(ts, CEC_TIM_DATA_BIT_1_HIGH);
|
|
else
|
|
ts = ktime_add_us(ts, CEC_TIM_DATA_BIT_0_HIGH);
|
|
}
|
|
}
|
|
|
|
static void vivid_cec_pin_events(struct vivid_dev *dev,
|
|
const struct cec_msg *msg, bool nacked)
|
|
{
|
|
ktime_t ts = ktime_get();
|
|
unsigned int i;
|
|
|
|
vivid_cec_pin_adap_events(dev->cec_rx_adap, ts, msg, nacked);
|
|
for (i = 0; i < MAX_OUTPUTS; i++)
|
|
vivid_cec_pin_adap_events(dev->cec_tx_adap[i], ts, msg, nacked);
|
|
}
|
|
|
|
static void vivid_cec_xfer_done_worker(struct work_struct *work)
|
|
{
|
|
struct vivid_cec_work *cw =
|
|
container_of(work, struct vivid_cec_work, work.work);
|
|
struct vivid_dev *dev = cw->dev;
|
|
struct cec_adapter *adap = cw->adap;
|
|
u8 dest = cec_msg_destination(&cw->msg);
|
|
bool valid_dest;
|
|
unsigned int i;
|
|
|
|
valid_dest = cec_msg_is_broadcast(&cw->msg);
|
|
if (!valid_dest)
|
|
valid_dest = vivid_cec_find_dest_adap(dev, adap, dest);
|
|
|
|
cw->tx_status = valid_dest ? CEC_TX_STATUS_OK : CEC_TX_STATUS_NACK;
|
|
spin_lock(&dev->cec_slock);
|
|
dev->cec_xfer_time_jiffies = 0;
|
|
dev->cec_xfer_start_jiffies = 0;
|
|
list_del(&cw->list);
|
|
spin_unlock(&dev->cec_slock);
|
|
vivid_cec_pin_events(dev, &cw->msg, !valid_dest);
|
|
cec_transmit_attempt_done(cw->adap, cw->tx_status);
|
|
|
|
/* Broadcast message */
|
|
if (adap != dev->cec_rx_adap)
|
|
cec_received_msg(dev->cec_rx_adap, &cw->msg);
|
|
for (i = 0; i < MAX_OUTPUTS && dev->cec_tx_adap[i]; i++)
|
|
if (adap != dev->cec_tx_adap[i])
|
|
cec_received_msg(dev->cec_tx_adap[i], &cw->msg);
|
|
kfree(cw);
|
|
}
|
|
|
|
static void vivid_cec_xfer_try_worker(struct work_struct *work)
|
|
{
|
|
struct vivid_cec_work *cw =
|
|
container_of(work, struct vivid_cec_work, work.work);
|
|
struct vivid_dev *dev = cw->dev;
|
|
|
|
spin_lock(&dev->cec_slock);
|
|
if (dev->cec_xfer_time_jiffies) {
|
|
list_del(&cw->list);
|
|
spin_unlock(&dev->cec_slock);
|
|
cec_transmit_attempt_done(cw->adap, CEC_TX_STATUS_ARB_LOST);
|
|
kfree(cw);
|
|
} else {
|
|
INIT_DELAYED_WORK(&cw->work, vivid_cec_xfer_done_worker);
|
|
dev->cec_xfer_start_jiffies = jiffies;
|
|
dev->cec_xfer_time_jiffies = usecs_to_jiffies(cw->usecs);
|
|
spin_unlock(&dev->cec_slock);
|
|
schedule_delayed_work(&cw->work, dev->cec_xfer_time_jiffies);
|
|
}
|
|
}
|
|
|
|
static int vivid_cec_adap_enable(struct cec_adapter *adap, bool enable)
|
|
{
|
|
adap->cec_pin_is_high = true;
|
|
return 0;
|
|
}
|
|
|
|
static int vivid_cec_adap_log_addr(struct cec_adapter *adap, u8 log_addr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* One data bit takes 2400 us, each byte needs 10 bits so that's 24000 us
|
|
* per byte.
|
|
*/
|
|
#define USECS_PER_BYTE 24000
|
|
|
|
static int vivid_cec_adap_transmit(struct cec_adapter *adap, u8 attempts,
|
|
u32 signal_free_time, struct cec_msg *msg)
|
|
{
|
|
struct vivid_dev *dev = cec_get_drvdata(adap);
|
|
struct vivid_cec_work *cw = kzalloc(sizeof(*cw), GFP_KERNEL);
|
|
long delta_jiffies = 0;
|
|
|
|
if (cw == NULL)
|
|
return -ENOMEM;
|
|
cw->dev = dev;
|
|
cw->adap = adap;
|
|
cw->usecs = CEC_FREE_TIME_TO_USEC(signal_free_time) +
|
|
msg->len * USECS_PER_BYTE;
|
|
cw->msg = *msg;
|
|
|
|
spin_lock(&dev->cec_slock);
|
|
list_add(&cw->list, &dev->cec_work_list);
|
|
if (dev->cec_xfer_time_jiffies == 0) {
|
|
INIT_DELAYED_WORK(&cw->work, vivid_cec_xfer_done_worker);
|
|
dev->cec_xfer_start_jiffies = jiffies;
|
|
dev->cec_xfer_time_jiffies = usecs_to_jiffies(cw->usecs);
|
|
delta_jiffies = dev->cec_xfer_time_jiffies;
|
|
} else {
|
|
INIT_DELAYED_WORK(&cw->work, vivid_cec_xfer_try_worker);
|
|
delta_jiffies = dev->cec_xfer_start_jiffies +
|
|
dev->cec_xfer_time_jiffies - jiffies;
|
|
}
|
|
spin_unlock(&dev->cec_slock);
|
|
schedule_delayed_work(&cw->work, delta_jiffies < 0 ? 0 : delta_jiffies);
|
|
return 0;
|
|
}
|
|
|
|
static int vivid_received(struct cec_adapter *adap, struct cec_msg *msg)
|
|
{
|
|
struct vivid_dev *dev = cec_get_drvdata(adap);
|
|
struct cec_msg reply;
|
|
u8 dest = cec_msg_destination(msg);
|
|
u8 disp_ctl;
|
|
char osd[14];
|
|
|
|
if (cec_msg_is_broadcast(msg))
|
|
dest = adap->log_addrs.log_addr[0];
|
|
cec_msg_init(&reply, dest, cec_msg_initiator(msg));
|
|
|
|
switch (cec_msg_opcode(msg)) {
|
|
case CEC_MSG_SET_OSD_STRING:
|
|
if (!cec_is_sink(adap))
|
|
return -ENOMSG;
|
|
cec_ops_set_osd_string(msg, &disp_ctl, osd);
|
|
switch (disp_ctl) {
|
|
case CEC_OP_DISP_CTL_DEFAULT:
|
|
strcpy(dev->osd, osd);
|
|
dev->osd_jiffies = jiffies;
|
|
break;
|
|
case CEC_OP_DISP_CTL_UNTIL_CLEARED:
|
|
strcpy(dev->osd, osd);
|
|
dev->osd_jiffies = 0;
|
|
break;
|
|
case CEC_OP_DISP_CTL_CLEAR:
|
|
dev->osd[0] = 0;
|
|
dev->osd_jiffies = 0;
|
|
break;
|
|
default:
|
|
cec_msg_feature_abort(&reply, cec_msg_opcode(msg),
|
|
CEC_OP_ABORT_INVALID_OP);
|
|
cec_transmit_msg(adap, &reply, false);
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
return -ENOMSG;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static const struct cec_adap_ops vivid_cec_adap_ops = {
|
|
.adap_enable = vivid_cec_adap_enable,
|
|
.adap_log_addr = vivid_cec_adap_log_addr,
|
|
.adap_transmit = vivid_cec_adap_transmit,
|
|
.received = vivid_received,
|
|
};
|
|
|
|
struct cec_adapter *vivid_cec_alloc_adap(struct vivid_dev *dev,
|
|
unsigned int idx,
|
|
bool is_source)
|
|
{
|
|
char name[sizeof(dev->vid_out_dev.name) + 2];
|
|
u32 caps = CEC_CAP_DEFAULTS | CEC_CAP_MONITOR_ALL | CEC_CAP_MONITOR_PIN;
|
|
|
|
snprintf(name, sizeof(name), "%s%d",
|
|
is_source ? dev->vid_out_dev.name : dev->vid_cap_dev.name,
|
|
idx);
|
|
return cec_allocate_adapter(&vivid_cec_adap_ops, dev,
|
|
name, caps, 1);
|
|
}
|