2021-01-07 14:28:38 +01:00
/* SPDX-License-Identifier: GPL-2.0 */
/* Author: Dan Scally <djrscally@gmail.com> */
2023-05-18 12:05:21 +02:00
# ifndef __IPU_BRIDGE_H
# define __IPU_BRIDGE_H
2021-01-07 14:28:38 +01:00
# include <linux/property.h>
# include <linux/types.h>
2023-07-05 23:30:00 +02:00
# include <media/v4l2-fwnode.h>
2021-01-07 14:28:38 +01:00
2023-05-18 12:05:21 +02:00
# define IPU_HID "INT343E"
# define IPU_MAX_LANES 4
2023-05-18 12:05:22 +02:00
# define IPU_MAX_PORTS 4
2021-01-07 14:28:38 +01:00
# define MAX_NUM_LINK_FREQS 3
2021-07-12 11:03:26 +02:00
/* Values are educated guesses as we don't have a spec */
2023-05-18 12:05:21 +02:00
# define IPU_SENSOR_ROTATION_NORMAL 0
# define IPU_SENSOR_ROTATION_INVERTED 1
2021-07-12 11:03:26 +02:00
2023-05-18 12:05:21 +02:00
# define IPU_SENSOR_CONFIG(_HID, _NR, ...) \
( const struct ipu_sensor_config ) { \
2021-01-07 14:28:38 +01:00
. hid = _HID , \
. nr_link_freqs = _NR , \
. link_freqs = { __VA_ARGS__ } \
}
# define NODE_SENSOR(_HID, _PROPS) \
( const struct software_node ) { \
. name = _HID , \
. properties = _PROPS , \
}
# define NODE_PORT(_PORT, _SENSOR_NODE) \
( const struct software_node ) { \
. name = _PORT , \
. parent = _SENSOR_NODE , \
}
# define NODE_ENDPOINT(_EP, _PORT, _PROPS) \
( const struct software_node ) { \
. name = _EP , \
. parent = _PORT , \
. properties = _PROPS , \
}
2021-12-03 11:28:57 +01:00
# define NODE_VCM(_TYPE) \
( const struct software_node ) { \
. name = _TYPE , \
}
2023-05-18 12:05:21 +02:00
enum ipu_sensor_swnodes {
2021-01-07 14:28:38 +01:00
SWNODE_SENSOR_HID ,
SWNODE_SENSOR_PORT ,
SWNODE_SENSOR_ENDPOINT ,
2023-05-18 12:05:21 +02:00
SWNODE_IPU_PORT ,
SWNODE_IPU_ENDPOINT ,
2023-08-03 07:38:14 +02:00
/* below are optional / maybe empty */
SWNODE_IVSC_HID ,
SWNODE_IVSC_SENSOR_PORT ,
SWNODE_IVSC_SENSOR_ENDPOINT ,
SWNODE_IVSC_IPU_PORT ,
SWNODE_IVSC_IPU_ENDPOINT ,
2021-12-03 11:28:57 +01:00
SWNODE_VCM ,
2021-01-07 14:28:38 +01:00
SWNODE_COUNT
} ;
/* Data representation as it is in ACPI SSDB buffer */
2023-05-18 12:05:21 +02:00
struct ipu_sensor_ssdb {
2021-01-07 14:28:38 +01:00
u8 version ;
u8 sku ;
u8 guid_csi2 [ 16 ] ;
u8 devfunction ;
u8 bus ;
u32 dphylinkenfuses ;
u32 clockdiv ;
u8 link ;
u8 lanes ;
u32 csiparams [ 10 ] ;
u32 maxlanespeed ;
u8 sensorcalibfileidx ;
u8 sensorcalibfileidxInMBZ [ 3 ] ;
u8 romtype ;
u8 vcmtype ;
u8 platforminfo ;
u8 platformsubinfo ;
u8 flash ;
u8 privacyled ;
u8 degree ;
u8 mipilinkdefined ;
u32 mclkspeed ;
u8 controllogicid ;
u8 reserved1 [ 3 ] ;
u8 mclkport ;
u8 reserved2 [ 13 ] ;
} __packed ;
2023-05-18 12:05:21 +02:00
struct ipu_property_names {
2021-01-07 14:28:38 +01:00
char clock_frequency [ 16 ] ;
char rotation [ 9 ] ;
2021-07-12 11:03:26 +02:00
char orientation [ 12 ] ;
2021-01-07 14:28:38 +01:00
char bus_type [ 9 ] ;
char data_lanes [ 11 ] ;
char remote_endpoint [ 16 ] ;
char link_frequencies [ 17 ] ;
} ;
2023-05-18 12:05:21 +02:00
struct ipu_node_names {
2021-01-07 14:28:38 +01:00
char port [ 7 ] ;
2023-08-03 07:38:14 +02:00
char ivsc_sensor_port [ 7 ] ;
char ivsc_ipu_port [ 7 ] ;
2021-01-07 14:28:38 +01:00
char endpoint [ 11 ] ;
char remote_port [ 7 ] ;
2023-07-05 23:29:54 +02:00
char vcm [ 16 ] ;
2021-01-07 14:28:38 +01:00
} ;
2023-05-18 12:05:21 +02:00
struct ipu_sensor_config {
2021-01-07 14:28:38 +01:00
const char * hid ;
const u8 nr_link_freqs ;
const u64 link_freqs [ MAX_NUM_LINK_FREQS ] ;
} ;
2023-05-18 12:05:21 +02:00
struct ipu_sensor {
2023-03-10 23:19:10 +08:00
/* append ssdb.link(u8) in "-%u" format as suffix of HID */
char name [ ACPI_ID_LEN + 4 ] ;
2021-01-07 14:28:38 +01:00
struct acpi_device * adev ;
2023-08-03 07:38:14 +02:00
struct device * csi_dev ;
struct acpi_device * ivsc_adev ;
char ivsc_name [ ACPI_ID_LEN + 4 ] ;
2022-12-28 11:49:20 +02:00
/* SWNODE_COUNT + 1 for terminating NULL */
const struct software_node * group [ SWNODE_COUNT + 1 ] ;
struct software_node swnodes [ SWNODE_COUNT ] ;
2023-05-18 12:05:21 +02:00
struct ipu_node_names node_names ;
2021-01-07 14:28:38 +01:00
2023-07-05 23:30:00 +02:00
u8 link ;
u8 lanes ;
u32 mclkspeed ;
u32 rotation ;
enum v4l2_fwnode_orientation orientation ;
const char * vcm_type ;
2021-07-12 11:03:26 +02:00
2023-05-18 12:05:21 +02:00
struct ipu_property_names prop_names ;
2021-01-07 14:28:38 +01:00
struct property_entry ep_properties [ 5 ] ;
2021-12-03 11:28:57 +01:00
struct property_entry dev_properties [ 5 ] ;
2023-05-18 12:05:21 +02:00
struct property_entry ipu_properties [ 3 ] ;
2023-08-03 07:38:14 +02:00
struct property_entry ivsc_properties [ 1 ] ;
struct property_entry ivsc_sensor_ep_properties [ 4 ] ;
struct property_entry ivsc_ipu_ep_properties [ 4 ] ;
2021-01-07 14:28:38 +01:00
struct software_node_ref_args local_ref [ 1 ] ;
struct software_node_ref_args remote_ref [ 1 ] ;
2021-12-03 11:28:57 +01:00
struct software_node_ref_args vcm_ref [ 1 ] ;
2023-08-03 07:38:14 +02:00
struct software_node_ref_args ivsc_sensor_ref [ 1 ] ;
struct software_node_ref_args ivsc_ipu_ref [ 1 ] ;
2021-01-07 14:28:38 +01:00
} ;
2023-07-05 23:30:02 +02:00
typedef int ( * ipu_parse_sensor_fwnode_t ) ( struct acpi_device * adev ,
struct ipu_sensor * sensor ) ;
2023-05-18 12:05:21 +02:00
struct ipu_bridge {
2023-07-05 23:29:58 +02:00
struct device * dev ;
2023-07-05 23:30:02 +02:00
ipu_parse_sensor_fwnode_t parse_sensor_fwnode ;
2023-05-18 12:05:21 +02:00
char ipu_node_name [ ACPI_ID_LEN ] ;
struct software_node ipu_hid_node ;
2021-01-07 14:28:38 +01:00
u32 data_lanes [ 4 ] ;
unsigned int n_sensors ;
2023-05-18 12:05:22 +02:00
struct ipu_sensor sensors [ IPU_MAX_PORTS ] ;
2021-01-07 14:28:38 +01:00
} ;
2023-05-18 12:05:21 +02:00
# if IS_ENABLED(CONFIG_IPU_BRIDGE)
2023-07-05 23:30:02 +02:00
int ipu_bridge_init ( struct device * dev ,
ipu_parse_sensor_fwnode_t parse_sensor_fwnode ) ;
int ipu_bridge_parse_ssdb ( struct acpi_device * adev , struct ipu_sensor * sensor ) ;
2023-07-05 23:30:05 +02:00
int ipu_bridge_instantiate_vcm ( struct device * sensor ) ;
2023-05-18 12:05:21 +02:00
# else
2023-07-05 23:30:02 +02:00
/* Use a define to avoid the @parse_sensor_fwnode argument getting evaluated */
# define ipu_bridge_init(dev, parse_sensor_fwnode) (0)
2023-07-05 23:30:05 +02:00
static inline int ipu_bridge_instantiate_vcm ( struct device * s ) { return 0 ; }
2023-05-18 12:05:21 +02:00
# endif
2021-01-07 14:28:38 +01:00
# endif