2016-06-25 09:43:58 -03:00
/*
* cec - HDMI Consumer Electronics Control support header
*
* Copyright 2016 Cisco Systems , Inc . and / or its affiliates . All rights reserved .
*
* This program is free software ; you may redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; version 2 of the License .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND ,
* EXPRESS OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY , FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT . IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER LIABILITY , WHETHER IN AN
* ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM , OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE .
*/
# ifndef _MEDIA_CEC_H
# define _MEDIA_CEC_H
# include <linux/poll.h>
# include <linux/fs.h>
# include <linux/debugfs.h>
# include <linux/device.h>
# include <linux/cdev.h>
# include <linux/kthread.h>
# include <linux/timer.h>
# include <linux/cec-funcs.h>
# include <media/rc-core.h>
2016-12-13 12:15:57 -02:00
# include <media/cec-notifier.h>
2016-06-25 09:43:58 -03:00
/**
* struct cec_devnode - cec device node
* @ dev : cec device
* @ cdev : cec character device
* @ minor : device node minor number
* @ registered : the device was correctly registered
* @ unregistered : the device was unregistered
* @ fhs_lock : lock to control access to the filehandle list
* @ fhs : the list of open filehandles ( cec_fh )
*
* This structure represents a cec - related device node .
*
* The @ parent is a physical device . It must be set by core or device drivers
* before registering the node .
*/
struct cec_devnode {
/* sysfs */
struct device dev ;
struct cdev cdev ;
/* device info */
int minor ;
bool registered ;
bool unregistered ;
struct list_head fhs ;
2016-08-02 08:11:00 -03:00
struct mutex lock ;
2016-06-25 09:43:58 -03:00
} ;
struct cec_adapter ;
struct cec_data ;
struct cec_data {
struct list_head list ;
struct list_head xfer_list ;
struct cec_adapter * adap ;
struct cec_msg msg ;
struct cec_fh * fh ;
struct delayed_work work ;
struct completion c ;
u8 attempts ;
bool new_initiator ;
bool blocking ;
bool completed ;
} ;
struct cec_msg_entry {
struct list_head list ;
struct cec_msg msg ;
} ;
2017-07-11 03:30:39 -03:00
struct cec_event_entry {
struct list_head list ;
struct cec_event ev ;
} ;
# define CEC_NUM_CORE_EVENTS 2
# define CEC_NUM_EVENTS CEC_EVENT_PIN_HIGH
2016-06-25 09:43:58 -03:00
struct cec_fh {
struct list_head list ;
struct list_head xfer_list ;
struct cec_adapter * adap ;
u8 mode_initiator ;
u8 mode_follower ;
/* Events */
wait_queue_head_t wait ;
struct mutex lock ;
2017-07-11 03:30:39 -03:00
struct list_head events [ CEC_NUM_EVENTS ] ; /* queued events */
u8 queued_events [ CEC_NUM_EVENTS ] ;
unsigned int total_queued_events ;
struct cec_event_entry core_events [ CEC_NUM_CORE_EVENTS ] ;
2016-06-25 09:43:58 -03:00
struct list_head msgs ; /* queued messages */
unsigned int queued_msgs ;
} ;
# define CEC_SIGNAL_FREE_TIME_RETRY 3
# define CEC_SIGNAL_FREE_TIME_NEW_INITIATOR 5
# define CEC_SIGNAL_FREE_TIME_NEXT_XFER 7
/* The nominal data bit period is 2.4 ms */
# define CEC_FREE_TIME_TO_USEC(ft) ((ft) * 2400)
struct cec_adap_ops {
/* Low-level callbacks */
int ( * adap_enable ) ( struct cec_adapter * adap , bool enable ) ;
int ( * adap_monitor_all_enable ) ( struct cec_adapter * adap , bool enable ) ;
int ( * adap_log_addr ) ( struct cec_adapter * adap , u8 logical_addr ) ;
int ( * adap_transmit ) ( struct cec_adapter * adap , u8 attempts ,
u32 signal_free_time , struct cec_msg * msg ) ;
void ( * adap_status ) ( struct cec_adapter * adap , struct seq_file * file ) ;
2017-07-11 03:30:36 -03:00
void ( * adap_free ) ( struct cec_adapter * adap ) ;
2016-06-25 09:43:58 -03:00
/* High-level CEC message callback */
int ( * received ) ( struct cec_adapter * adap , struct cec_msg * msg ) ;
} ;
/*
* The minimum message length you can receive ( excepting poll messages ) is 2.
* With a transfer rate of at most 36 bytes per second this makes 18 messages
* per second worst case .
*
2016-07-17 11:40:05 -03:00
* We queue at most 3 seconds worth of received messages . The CEC specification
* requires that messages are replied to within a second , so 3 seconds should
* give more than enough margin . Since most messages are actually more than 2
* bytes , this is in practice a lot more than 3 seconds .
2016-06-25 09:43:58 -03:00
*/
2016-07-17 11:40:05 -03:00
# define CEC_MAX_MSG_RX_QUEUE_SZ (18 * 3)
/*
* The transmit queue is limited to 1 second worth of messages ( worst case ) .
* Messages can be transmitted by userspace and kernel space . But for both it
* makes no sense to have a lot of messages queued up . One second seems
* reasonable .
*/
# define CEC_MAX_MSG_TX_QUEUE_SZ (18 * 1)
2016-06-25 09:43:58 -03:00
struct cec_adapter {
struct module * owner ;
char name [ 32 ] ;
struct cec_devnode devnode ;
struct mutex lock ;
struct rc_dev * rc ;
struct list_head transmit_queue ;
2016-07-17 11:40:05 -03:00
unsigned int transmit_queue_sz ;
2016-06-25 09:43:58 -03:00
struct list_head wait_queue ;
struct cec_data * transmitting ;
struct task_struct * kthread_config ;
struct completion config_completion ;
struct task_struct * kthread ;
wait_queue_head_t kthread_waitq ;
wait_queue_head_t waitq ;
const struct cec_adap_ops * ops ;
void * priv ;
u32 capabilities ;
u8 available_log_addrs ;
u16 phys_addr ;
2017-06-07 11:46:12 -03:00
bool needs_hpd ;
2016-06-25 09:43:58 -03:00
bool is_configuring ;
bool is_configured ;
u32 monitor_all_cnt ;
u32 follower_cnt ;
struct cec_fh * cec_follower ;
struct cec_fh * cec_initiator ;
bool passthrough ;
struct cec_log_addrs log_addrs ;
2017-07-11 03:30:34 -03:00
u32 tx_timeouts ;
2017-05-28 05:58:04 -03:00
# ifdef CONFIG_CEC_NOTIFIER
2016-12-13 12:15:57 -02:00
struct cec_notifier * notifier ;
# endif
2016-06-25 09:43:58 -03:00
struct dentry * cec_dir ;
struct dentry * status_file ;
u16 phys_addrs [ 15 ] ;
u32 sequence ;
char input_name [ 32 ] ;
char input_phys [ 32 ] ;
char input_drv [ 32 ] ;
} ;
2017-03-24 13:47:52 -03:00
static inline void * cec_get_drvdata ( const struct cec_adapter * adap )
{
return adap - > priv ;
}
2016-06-25 09:43:58 -03:00
static inline bool cec_has_log_addr ( const struct cec_adapter * adap , u8 log_addr )
{
return adap - > log_addrs . log_addr_mask & ( 1 < < log_addr ) ;
}
static inline bool cec_is_sink ( const struct cec_adapter * adap )
{
return adap - > phys_addr = = 0 ;
}
2017-04-17 07:54:37 -03:00
# define cec_phys_addr_exp(pa) \
( ( pa ) > > 12 ) , ( ( pa ) > > 8 ) & 0xf , ( ( pa ) > > 4 ) & 0xf , ( pa ) & 0xf
2017-06-07 11:46:08 -03:00
struct edid ;
2017-06-08 15:37:44 -03:00
# if IS_REACHABLE(CONFIG_CEC_CORE)
2016-06-25 09:43:58 -03:00
struct cec_adapter * cec_allocate_adapter ( const struct cec_adap_ops * ops ,
2016-11-25 06:23:34 -02:00
void * priv , const char * name , u32 caps , u8 available_las ) ;
int cec_register_adapter ( struct cec_adapter * adap , struct device * parent ) ;
2016-06-25 09:43:58 -03:00
void cec_unregister_adapter ( struct cec_adapter * adap ) ;
void cec_delete_adapter ( struct cec_adapter * adap ) ;
int cec_s_log_addrs ( struct cec_adapter * adap , struct cec_log_addrs * log_addrs ,
bool block ) ;
void cec_s_phys_addr ( struct cec_adapter * adap , u16 phys_addr ,
bool block ) ;
2017-06-07 11:46:08 -03:00
void cec_s_phys_addr_from_edid ( struct cec_adapter * adap ,
const struct edid * edid ) ;
2016-06-25 09:43:58 -03:00
int cec_transmit_msg ( struct cec_adapter * adap , struct cec_msg * msg ,
bool block ) ;
/* Called by the adapter */
2017-07-11 03:30:35 -03:00
void cec_transmit_done_ts ( struct cec_adapter * adap , u8 status ,
u8 arb_lost_cnt , u8 nack_cnt , u8 low_drive_cnt ,
u8 error_cnt , ktime_t ts ) ;
static inline void cec_transmit_done ( struct cec_adapter * adap , u8 status ,
u8 arb_lost_cnt , u8 nack_cnt ,
u8 low_drive_cnt , u8 error_cnt )
{
cec_transmit_done_ts ( adap , status , arb_lost_cnt , nack_cnt ,
low_drive_cnt , error_cnt , ktime_get ( ) ) ;
}
2017-06-07 11:46:10 -03:00
/*
* Simplified version of cec_transmit_done for hardware that doesn ' t retry
* failed transmits . So this is always just one attempt in which case
* the status is sufficient .
*/
2017-07-11 03:30:35 -03:00
void cec_transmit_attempt_done_ts ( struct cec_adapter * adap ,
u8 status , ktime_t ts ) ;
static inline void cec_transmit_attempt_done ( struct cec_adapter * adap ,
u8 status )
{
cec_transmit_attempt_done_ts ( adap , status , ktime_get ( ) ) ;
}
void cec_received_msg_ts ( struct cec_adapter * adap ,
struct cec_msg * msg , ktime_t ts ) ;
static inline void cec_received_msg ( struct cec_adapter * adap ,
struct cec_msg * msg )
{
cec_received_msg_ts ( adap , msg , ktime_get ( ) ) ;
}
2016-06-25 09:43:58 -03:00
2017-04-17 07:54:37 -03:00
/**
* cec_get_edid_phys_addr ( ) - find and return the physical address
*
* @ edid : pointer to the EDID data
* @ size : size in bytes of the EDID data
* @ offset : If not % NULL then the location of the physical address
* bytes in the EDID will be returned here . This is set to 0
* if there is no physical address found .
*
* Return : the physical address or CEC_PHYS_ADDR_INVALID if there is none .
*/
u16 cec_get_edid_phys_addr ( const u8 * edid , unsigned int size ,
unsigned int * offset ) ;
/**
* cec_set_edid_phys_addr ( ) - find and set the physical address
*
* @ edid : pointer to the EDID data
* @ size : size in bytes of the EDID data
* @ phys_addr : the new physical address
*
* This function finds the location of the physical address in the EDID
* and fills in the given physical address and updates the checksum
* at the end of the EDID block . It does nothing if the EDID doesn ' t
* contain a physical address .
*/
void cec_set_edid_phys_addr ( u8 * edid , unsigned int size , u16 phys_addr ) ;
/**
* cec_phys_addr_for_input ( ) - calculate the PA for an input
*
* @ phys_addr : the physical address of the parent
* @ input : the number of the input port , must be between 1 and 15
*
* This function calculates a new physical address based on the input
* port number . For example :
*
* PA = 0.0 .0 .0 and input = 2 becomes 2.0 .0 .0
*
* PA = 3.0 .0 .0 and input = 1 becomes 3.1 .0 .0
*
* PA = 3.2 .1 .0 and input = 5 becomes 3.2 .1 .5
*
* PA = 3.2 .1 .3 and input = 5 becomes f . f . f . f since it maxed out the depth .
*
* Return : the new physical address or CEC_PHYS_ADDR_INVALID .
*/
u16 cec_phys_addr_for_input ( u16 phys_addr , u8 input ) ;
/**
* cec_phys_addr_validate ( ) - validate a physical address from an EDID
*
* @ phys_addr : the physical address to validate
* @ parent : if not % NULL , then this is filled with the parents PA .
* @ port : if not % NULL , then this is filled with the input port .
*
* This validates a physical address as read from an EDID . If the
* PA is invalid ( such as 1.0 .1 .0 since ' 0 ' is only allowed at the end ) ,
* then it will return - EINVAL .
*
* The parent PA is passed into % parent and the input port is passed into
* % port . For example :
*
* PA = 0.0 .0 .0 : has parent 0.0 .0 .0 and input port 0.
*
* PA = 1.0 .0 .0 : has parent 0.0 .0 .0 and input port 1.
*
* PA = 3.2 .0 .0 : has parent 3.0 .0 .0 and input port 2.
*
* PA = f . f . f . f : has parent f . f . f . f and input port 0.
*
* Return : 0 if the PA is valid , - EINVAL if not .
*/
int cec_phys_addr_validate ( u16 phys_addr , u16 * parent , u16 * port ) ;
2017-05-28 05:58:04 -03:00
# ifdef CONFIG_CEC_NOTIFIER
2016-12-13 12:15:57 -02:00
void cec_register_cec_notifier ( struct cec_adapter * adap ,
struct cec_notifier * notifier ) ;
# endif
2016-06-25 09:43:58 -03:00
# else
2016-11-25 06:23:34 -02:00
static inline int cec_register_adapter ( struct cec_adapter * adap ,
struct device * parent )
2016-06-25 09:43:58 -03:00
{
return 0 ;
}
static inline void cec_unregister_adapter ( struct cec_adapter * adap )
{
}
static inline void cec_delete_adapter ( struct cec_adapter * adap )
{
}
static inline void cec_s_phys_addr ( struct cec_adapter * adap , u16 phys_addr ,
bool block )
{
}
2017-06-07 11:46:08 -03:00
static inline void cec_s_phys_addr_from_edid ( struct cec_adapter * adap ,
const struct edid * edid )
{
}
2017-04-17 07:54:37 -03:00
static inline u16 cec_get_edid_phys_addr ( const u8 * edid , unsigned int size ,
unsigned int * offset )
{
if ( offset )
* offset = 0 ;
return CEC_PHYS_ADDR_INVALID ;
}
static inline void cec_set_edid_phys_addr ( u8 * edid , unsigned int size ,
u16 phys_addr )
{
}
static inline u16 cec_phys_addr_for_input ( u16 phys_addr , u8 input )
{
return CEC_PHYS_ADDR_INVALID ;
}
static inline int cec_phys_addr_validate ( u16 phys_addr , u16 * parent , u16 * port )
{
return 0 ;
}
2016-06-25 09:43:58 -03:00
# endif
2017-06-07 11:46:09 -03:00
/**
* cec_phys_addr_invalidate ( ) - set the physical address to INVALID
*
* @ adap : the CEC adapter
*
* This is a simple helper function to invalidate the physical
* address .
*/
static inline void cec_phys_addr_invalidate ( struct cec_adapter * adap )
{
cec_s_phys_addr ( adap , CEC_PHYS_ADDR_INVALID , false ) ;
}
2016-06-25 09:43:58 -03:00
# endif /* _MEDIA_CEC_H */