Commit 533a3f7b authored by Hans Verkuil's avatar Hans Verkuil Committed by Mauro Carvalho Chehab

[media] cec: allow specific messages even when unconfigured

The CEC specifications explicitly allows you to send poll messages and
Image/Text View On messages to a TV, even when unconfigured (i.e. there is
no hotplug signal detected). Some TVs will pull the HPD low when switching
to another input, or when going into standby, but CEC should still be
allowed to wake up such a display.

Add support for sending messages with initiator 0xf (Unregistered) and
destination 0 (TV) when no physical address is present.

This also required another change: the CEC adapter has to stay enabled as
long as 1) the CEC device is configured or 2) at least one filehandle is open
for the CEC device.
Signed-off-by: default avatarHans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@s-opensource.com>
parent b39f93ef
...@@ -367,7 +367,6 @@ int cec_thread_func(void *_adap) ...@@ -367,7 +367,6 @@ int cec_thread_func(void *_adap)
*/ */
err = wait_event_interruptible_timeout(adap->kthread_waitq, err = wait_event_interruptible_timeout(adap->kthread_waitq,
kthread_should_stop() || kthread_should_stop() ||
(!adap->is_configured && !adap->is_configuring) ||
(!adap->transmitting && (!adap->transmitting &&
!list_empty(&adap->transmit_queue)), !list_empty(&adap->transmit_queue)),
msecs_to_jiffies(CEC_XFER_TIMEOUT_MS)); msecs_to_jiffies(CEC_XFER_TIMEOUT_MS));
...@@ -382,8 +381,7 @@ int cec_thread_func(void *_adap) ...@@ -382,8 +381,7 @@ int cec_thread_func(void *_adap)
mutex_lock(&adap->lock); mutex_lock(&adap->lock);
if ((!adap->is_configured && !adap->is_configuring) || if (kthread_should_stop()) {
kthread_should_stop()) {
cec_flush(adap); cec_flush(adap);
goto unlock; goto unlock;
} }
...@@ -414,6 +412,7 @@ int cec_thread_func(void *_adap) ...@@ -414,6 +412,7 @@ int cec_thread_func(void *_adap)
struct cec_data, list); struct cec_data, list);
list_del_init(&data->list); list_del_init(&data->list);
adap->transmit_queue_sz--; adap->transmit_queue_sz--;
/* Make this the current transmitting message */ /* Make this the current transmitting message */
adap->transmitting = data; adap->transmitting = data;
...@@ -647,7 +646,8 @@ int cec_transmit_msg_fh(struct cec_adapter *adap, struct cec_msg *msg, ...@@ -647,7 +646,8 @@ int cec_transmit_msg_fh(struct cec_adapter *adap, struct cec_msg *msg,
cec_msg_initiator(msg)); cec_msg_initiator(msg));
return -EINVAL; return -EINVAL;
} }
if (!adap->is_configured && !adap->is_configuring) if (!adap->is_configured && !adap->is_configuring &&
(msg->msg[0] != 0xf0 || msg->reply))
return -ENONET; return -ENONET;
if (adap->transmit_queue_sz >= CEC_MAX_MSG_TX_QUEUE_SZ) if (adap->transmit_queue_sz >= CEC_MAX_MSG_TX_QUEUE_SZ)
...@@ -696,6 +696,7 @@ int cec_transmit_msg_fh(struct cec_adapter *adap, struct cec_msg *msg, ...@@ -696,6 +696,7 @@ int cec_transmit_msg_fh(struct cec_adapter *adap, struct cec_msg *msg,
if (fh) if (fh)
list_add_tail(&data->xfer_list, &fh->xfer_list); list_add_tail(&data->xfer_list, &fh->xfer_list);
list_add_tail(&data->list, &adap->transmit_queue); list_add_tail(&data->list, &adap->transmit_queue);
adap->transmit_queue_sz++; adap->transmit_queue_sz++;
if (!adap->transmitting) if (!adap->transmitting)
...@@ -1121,6 +1122,7 @@ static void cec_adap_unconfigure(struct cec_adapter *adap) ...@@ -1121,6 +1122,7 @@ static void cec_adap_unconfigure(struct cec_adapter *adap)
adap->is_configuring = false; adap->is_configuring = false;
adap->is_configured = false; adap->is_configured = false;
memset(adap->phys_addrs, 0xff, sizeof(adap->phys_addrs)); memset(adap->phys_addrs, 0xff, sizeof(adap->phys_addrs));
cec_flush(adap);
wake_up_interruptible(&adap->kthread_waitq); wake_up_interruptible(&adap->kthread_waitq);
cec_post_state_event(adap); cec_post_state_event(adap);
} }
...@@ -1352,19 +1354,30 @@ void __cec_s_phys_addr(struct cec_adapter *adap, u16 phys_addr, bool block) ...@@ -1352,19 +1354,30 @@ void __cec_s_phys_addr(struct cec_adapter *adap, u16 phys_addr, bool block)
/* Disabling monitor all mode should always succeed */ /* Disabling monitor all mode should always succeed */
if (adap->monitor_all_cnt) if (adap->monitor_all_cnt)
WARN_ON(call_op(adap, adap_monitor_all_enable, false)); WARN_ON(call_op(adap, adap_monitor_all_enable, false));
WARN_ON(adap->ops->adap_enable(adap, false)); mutex_lock(&adap->devnode.lock);
if (list_empty(&adap->devnode.fhs))
WARN_ON(adap->ops->adap_enable(adap, false));
mutex_unlock(&adap->devnode.lock);
if (phys_addr == CEC_PHYS_ADDR_INVALID) if (phys_addr == CEC_PHYS_ADDR_INVALID)
return; return;
} }
if (adap->ops->adap_enable(adap, true)) mutex_lock(&adap->devnode.lock);
if (list_empty(&adap->devnode.fhs) &&
adap->ops->adap_enable(adap, true)) {
mutex_unlock(&adap->devnode.lock);
return; return;
}
if (adap->monitor_all_cnt && if (adap->monitor_all_cnt &&
call_op(adap, adap_monitor_all_enable, true)) { call_op(adap, adap_monitor_all_enable, true)) {
WARN_ON(adap->ops->adap_enable(adap, false)); if (list_empty(&adap->devnode.fhs))
WARN_ON(adap->ops->adap_enable(adap, false));
mutex_unlock(&adap->devnode.lock);
return; return;
} }
mutex_unlock(&adap->devnode.lock);
adap->phys_addr = phys_addr; adap->phys_addr = phys_addr;
cec_post_state_event(adap); cec_post_state_event(adap);
if (adap->log_addrs.num_log_addrs) if (adap->log_addrs.num_log_addrs)
......
...@@ -198,7 +198,9 @@ static long cec_transmit(struct cec_adapter *adap, struct cec_fh *fh, ...@@ -198,7 +198,9 @@ static long cec_transmit(struct cec_adapter *adap, struct cec_fh *fh,
return -EINVAL; return -EINVAL;
mutex_lock(&adap->lock); mutex_lock(&adap->lock);
if (!adap->is_configured) if (adap->is_configuring)
err = -ENONET;
else if (!adap->is_configured && (msg.msg[0] != 0xf0 || msg.reply))
err = -ENONET; err = -ENONET;
else if (cec_is_busy(adap, fh)) else if (cec_is_busy(adap, fh))
err = -EBUSY; err = -EBUSY;
...@@ -515,9 +517,18 @@ static int cec_open(struct inode *inode, struct file *filp) ...@@ -515,9 +517,18 @@ static int cec_open(struct inode *inode, struct file *filp)
return err; return err;
} }
mutex_lock(&devnode->lock);
if (list_empty(&devnode->fhs) &&
adap->phys_addr == CEC_PHYS_ADDR_INVALID) {
err = adap->ops->adap_enable(adap, true);
if (err) {
mutex_unlock(&devnode->lock);
kfree(fh);
return err;
}
}
filp->private_data = fh; filp->private_data = fh;
mutex_lock(&devnode->lock);
/* Queue up initial state events */ /* Queue up initial state events */
ev_state.state_change.phys_addr = adap->phys_addr; ev_state.state_change.phys_addr = adap->phys_addr;
ev_state.state_change.log_addr_mask = adap->log_addrs.log_addr_mask; ev_state.state_change.log_addr_mask = adap->log_addrs.log_addr_mask;
...@@ -551,6 +562,10 @@ static int cec_release(struct inode *inode, struct file *filp) ...@@ -551,6 +562,10 @@ static int cec_release(struct inode *inode, struct file *filp)
mutex_lock(&devnode->lock); mutex_lock(&devnode->lock);
list_del(&fh->list); list_del(&fh->list);
if (list_empty(&devnode->fhs) &&
adap->phys_addr == CEC_PHYS_ADDR_INVALID) {
WARN_ON(adap->ops->adap_enable(adap, false));
}
mutex_unlock(&devnode->lock); mutex_unlock(&devnode->lock);
/* Unhook pending transmits from this filehandle. */ /* Unhook pending transmits from this filehandle. */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment