
Adds support for new mink call to configure secure camera sessions. This new mink call takes in additional parameters to support the new domain-id based security scheme. The additional parameters are in the form of csid_hw_idx_mask, cdm_hw_idx_mask and vc_mask. These are in addition to the existing PHY idx and CPHY/DPHY lanes info. The introduction of this new mink call deprecates the existing SCM call used to service secure camera sessions. What this means is that on all subsequent versions of this driver, all secure camera sessions will be serviced by this mink call, which is able to accommodate older and newer targets, and their programming of different register sets. This design enables the cam_csiphy_notify_secure_mode wrapper to remain the same, with the underlying mink call made in the cam_compat layer, depending on camera driver version. CRs-Fixed: 3317248 Change-Id: I575f4b85097c81f047f398216d0190b249e6b200 Signed-off-by: Li Sha Lim <quic_lishlim@quicinc.com>
745 lines
18 KiB
C
745 lines
18 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (c) 2014-2021, The Linux Foundation. All rights reserved.
|
|
* Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
|
|
*/
|
|
|
|
#include <linux/dma-mapping.h>
|
|
#include <linux/dma-buf.h>
|
|
#include <linux/of_address.h>
|
|
#include <linux/slab.h>
|
|
|
|
#include <soc/qcom/rpmh.h>
|
|
|
|
#include "cam_compat.h"
|
|
#include "cam_debug_util.h"
|
|
#include "cam_cpas_api.h"
|
|
#include "camera_main.h"
|
|
#include "cam_eeprom_dev.h"
|
|
#include "cam_eeprom_core.h"
|
|
|
|
#if IS_ENABLED(CONFIG_USE_RPMH_DRV_API)
|
|
#define CAM_RSC_DRV_IDENTIFIER "cam_rsc"
|
|
|
|
const struct device *cam_cpas_get_rsc_dev_for_drv(uint32_t index)
|
|
{
|
|
const struct device *rsc_dev;
|
|
|
|
rsc_dev = rpmh_get_device(CAM_RSC_DRV_IDENTIFIER, index);
|
|
if (!rsc_dev) {
|
|
CAM_ERR(CAM_CPAS, "Invalid dev for index: %u", index);
|
|
return NULL;
|
|
}
|
|
|
|
return rsc_dev;
|
|
}
|
|
|
|
int cam_cpas_start_drv_for_dev(const struct device *dev)
|
|
{
|
|
int rc = 0;
|
|
|
|
if (!dev) {
|
|
CAM_ERR(CAM_CPAS, "Invalid dev for DRV enable");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = rpmh_drv_start(dev);
|
|
if (rc) {
|
|
CAM_ERR(CAM_CPAS, "[%s] Failed in DRV start", dev_name(dev));
|
|
return rc;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
int cam_cpas_stop_drv_for_dev(const struct device *dev)
|
|
{
|
|
int rc = 0;
|
|
|
|
if (!dev) {
|
|
CAM_ERR(CAM_CPAS, "Invalid dev for DRV disable");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = rpmh_drv_stop(dev);
|
|
if (rc) {
|
|
CAM_ERR(CAM_CPAS, "[%s] Failed in DRV stop", dev_name(dev));
|
|
return rc;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
int cam_cpas_drv_channel_switch_for_dev(const struct device *dev)
|
|
{
|
|
int rc = 0;
|
|
|
|
if (!dev) {
|
|
CAM_ERR(CAM_CPAS, "Invalid dev for DRV channel switch");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = rpmh_write_sleep_and_wake_no_child(dev);
|
|
if (rc) {
|
|
CAM_ERR(CAM_CPAS, "[%s] Failed in DRV channel switch", dev_name(dev));
|
|
return rc;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
#else
|
|
const struct device *cam_cpas_get_rsc_dev_for_drv(uint32_t index)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
int cam_cpas_start_drv_for_dev(const struct device *dev)
|
|
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int cam_cpas_stop_drv_for_dev(const struct device *dev)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int cam_cpas_drv_channel_switch_for_dev(const struct device *dev)
|
|
{
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 4, 0)
|
|
int cam_reserve_icp_fw(struct cam_fw_alloc_info *icp_fw, size_t fw_length)
|
|
{
|
|
int rc = 0;
|
|
struct device_node *of_node;
|
|
struct device_node *mem_node;
|
|
struct resource res;
|
|
|
|
of_node = (icp_fw->fw_dev)->of_node;
|
|
mem_node = of_parse_phandle(of_node, "memory-region", 0);
|
|
if (!mem_node) {
|
|
rc = -ENOMEM;
|
|
CAM_ERR(CAM_SMMU, "FW memory carveout not found");
|
|
goto end;
|
|
}
|
|
rc = of_address_to_resource(mem_node, 0, &res);
|
|
of_node_put(mem_node);
|
|
if (rc < 0) {
|
|
CAM_ERR(CAM_SMMU, "Unable to get start of FW mem carveout");
|
|
goto end;
|
|
}
|
|
icp_fw->fw_hdl = res.start;
|
|
icp_fw->fw_kva = ioremap_wc(icp_fw->fw_hdl, fw_length);
|
|
if (!icp_fw->fw_kva) {
|
|
CAM_ERR(CAM_SMMU, "Failed to map the FW.");
|
|
rc = -ENOMEM;
|
|
goto end;
|
|
}
|
|
memset_io(icp_fw->fw_kva, 0, fw_length);
|
|
|
|
end:
|
|
return rc;
|
|
}
|
|
|
|
void cam_unreserve_icp_fw(struct cam_fw_alloc_info *icp_fw, size_t fw_length)
|
|
{
|
|
iounmap(icp_fw->fw_kva);
|
|
}
|
|
|
|
int cam_ife_notify_safe_lut_scm(bool safe_trigger)
|
|
{
|
|
const uint32_t smmu_se_ife = 0;
|
|
uint32_t camera_hw_version, rc = 0;
|
|
|
|
rc = cam_cpas_get_cpas_hw_version(&camera_hw_version);
|
|
if (!rc && qcom_scm_smmu_notify_secure_lut(smmu_se_ife, safe_trigger)) {
|
|
switch (camera_hw_version) {
|
|
case CAM_CPAS_TITAN_170_V100:
|
|
case CAM_CPAS_TITAN_170_V110:
|
|
case CAM_CPAS_TITAN_175_V100:
|
|
CAM_ERR(CAM_ISP, "scm call to enable safe failed");
|
|
rc = -EINVAL;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
void cam_cpastop_scm_write(struct cam_cpas_hw_errata_wa *errata_wa)
|
|
{
|
|
int reg_val;
|
|
|
|
qcom_scm_io_readl(errata_wa->data.reg_info.offset, ®_val);
|
|
reg_val |= errata_wa->data.reg_info.value;
|
|
qcom_scm_io_writel(errata_wa->data.reg_info.offset, reg_val);
|
|
}
|
|
|
|
static int camera_platform_compare_dev(struct device *dev, const void *data)
|
|
{
|
|
return platform_bus_type.match(dev, (struct device_driver *) data);
|
|
}
|
|
|
|
static int camera_i2c_compare_dev(struct device *dev, const void *data)
|
|
{
|
|
return i2c_bus_type.match(dev, (struct device_driver *) data);
|
|
}
|
|
#else
|
|
int cam_reserve_icp_fw(struct cam_fw_alloc_info *icp_fw, size_t fw_length)
|
|
{
|
|
int rc = 0;
|
|
|
|
icp_fw->fw_kva = dma_alloc_coherent(icp_fw->fw_dev, fw_length,
|
|
&icp_fw->fw_hdl, GFP_KERNEL);
|
|
|
|
if (!icp_fw->fw_kva) {
|
|
CAM_ERR(CAM_SMMU, "FW memory alloc failed");
|
|
rc = -ENOMEM;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
void cam_unreserve_icp_fw(struct cam_fw_alloc_info *icp_fw, size_t fw_length)
|
|
{
|
|
dma_free_coherent(icp_fw->fw_dev, fw_length, icp_fw->fw_kva,
|
|
icp_fw->fw_hdl);
|
|
}
|
|
|
|
int cam_ife_notify_safe_lut_scm(bool safe_trigger)
|
|
{
|
|
const uint32_t smmu_se_ife = 0;
|
|
uint32_t camera_hw_version, rc = 0;
|
|
struct scm_desc description = {
|
|
.arginfo = SCM_ARGS(2, SCM_VAL, SCM_VAL),
|
|
.args[0] = smmu_se_ife,
|
|
.args[1] = safe_trigger,
|
|
};
|
|
|
|
rc = cam_cpas_get_cpas_hw_version(&camera_hw_version);
|
|
if (!rc && scm_call2(SCM_SIP_FNID(0x15, 0x3), &description)) {
|
|
switch (camera_hw_version) {
|
|
case CAM_CPAS_TITAN_170_V100:
|
|
case CAM_CPAS_TITAN_170_V110:
|
|
case CAM_CPAS_TITAN_175_V100:
|
|
CAM_ERR(CAM_ISP, "scm call to enable safe failed");
|
|
rc = -EINVAL;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
void cam_cpastop_scm_write(struct cam_cpas_hw_errata_wa *errata_wa)
|
|
{
|
|
int reg_val;
|
|
|
|
reg_val = scm_io_read(errata_wa->data.reg_info.offset);
|
|
reg_val |= errata_wa->data.reg_info.value;
|
|
scm_io_write(errata_wa->data.reg_info.offset, reg_val);
|
|
}
|
|
|
|
static int camera_platform_compare_dev(struct device *dev, void *data)
|
|
{
|
|
return platform_bus_type.match(dev, (struct device_driver *) data);
|
|
}
|
|
|
|
static int camera_i2c_compare_dev(struct device *dev, void *data)
|
|
{
|
|
return i2c_bus_type.match(dev, (struct device_driver *) data);
|
|
}
|
|
#endif
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 10, 0)
|
|
void cam_free_clear(const void * ptr)
|
|
{
|
|
kfree_sensitive(ptr);
|
|
}
|
|
#else
|
|
void cam_free_clear(const void * ptr)
|
|
{
|
|
kzfree(ptr);
|
|
}
|
|
#endif
|
|
|
|
#if KERNEL_VERSION(6, 0, 0) <= LINUX_VERSION_CODE
|
|
int cam_csiphy_notify_secure_mode(struct csiphy_device *csiphy_dev,
|
|
bool protect, int32_t offset)
|
|
{
|
|
int rc = 0;
|
|
struct Object client_env, sc_object;
|
|
ITCDriverSensorInfo params = {0};
|
|
struct cam_csiphy_secure_info *secure_info;
|
|
|
|
if (offset >= CSIPHY_MAX_INSTANCES_PER_PHY) {
|
|
CAM_ERR(CAM_CSIPHY, "Invalid CSIPHY offset");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = get_client_env_object(&client_env);
|
|
if (rc) {
|
|
CAM_ERR(CAM_CSIPHY, "Failed getting mink env object, rc: %d", rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = IClientEnv_open(client_env, CTrustedCameraDriver_UID, &sc_object);
|
|
if (rc) {
|
|
CAM_ERR(CAM_CSIPHY, "Failed getting mink sc_object, rc: %d", rc);
|
|
return rc;
|
|
}
|
|
|
|
secure_info = &csiphy_dev->csiphy_info[offset].secure_info;
|
|
params.csid_hw_idx_mask = secure_info->csid_hw_idx_mask;
|
|
params.cdm_hw_idx_mask = secure_info->cdm_hw_idx_mask;
|
|
params.vc_mask = secure_info->vc_mask;
|
|
params.phy_lane_sel_mask =
|
|
csiphy_dev->csiphy_info[offset].csiphy_phy_lane_sel_mask;
|
|
params.protect = protect ? 1 : 0;
|
|
|
|
rc = ITrustedCameraDriver_dynamicProtectSensor(sc_object, ¶ms);
|
|
if (rc) {
|
|
CAM_ERR(CAM_CSIPHY, "Mink secure call failed, rc: %d", rc);
|
|
return rc;
|
|
}
|
|
|
|
rc = Object_release(sc_object);
|
|
if (rc) {
|
|
CAM_ERR(CAM_CSIPHY, "Failed releasing secure camera object, rc: %d", rc);
|
|
return rc;
|
|
}
|
|
rc = Object_release(client_env);
|
|
if (rc) {
|
|
CAM_ERR(CAM_CSIPHY, "Failed releasing mink env object, rc: %d", rc);
|
|
return rc;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#elif KERNEL_VERSION(5, 4, 0) <= LINUX_VERSION_CODE
|
|
int cam_csiphy_notify_secure_mode(struct csiphy_device *csiphy_dev,
|
|
bool protect, int32_t offset)
|
|
{
|
|
int rc = 0;
|
|
|
|
/**
|
|
* A check here is made if the target is using
|
|
* an older version of the kernel driver (< 6.0)
|
|
* with domain id feature present. In this case,
|
|
* we are to fail this call, as the new mink call
|
|
* is only supported on kernel driver versions 6.0
|
|
* and above, and the new domain id scheme is not
|
|
* backwards compatible with the older scheme.
|
|
*/
|
|
if (csiphy_dev->domain_id_security) {
|
|
CAM_ERR(CAM_CSIPHY,
|
|
"Domain id support not present on current kernel driver: %d",
|
|
LINUX_VERSION_CODE);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (offset >= CSIPHY_MAX_INSTANCES_PER_PHY) {
|
|
CAM_ERR(CAM_CSIPHY, "Invalid CSIPHY offset");
|
|
rc = -EINVAL;
|
|
} else if (qcom_scm_camera_protect_phy_lanes(protect,
|
|
csiphy_dev->csiphy_info[offset].csiphy_cpas_cp_reg_mask)) {
|
|
CAM_ERR(CAM_CSIPHY, "SCM call to hypervisor failed");
|
|
rc = -EINVAL;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
#else
|
|
int cam_csiphy_notify_secure_mode(struct csiphy_device *csiphy_dev,
|
|
bool protect, int32_t offset)
|
|
{
|
|
int rc = 0;
|
|
struct scm_desc description = {
|
|
.arginfo = SCM_ARGS(2, SCM_VAL, SCM_VAL),
|
|
.args[0] = protect,
|
|
.args[1] = csiphy_dev->csiphy_info[offset]
|
|
.csiphy_cpas_cp_reg_mask,
|
|
};
|
|
|
|
if (offset >= CSIPHY_MAX_INSTANCES_PER_PHY) {
|
|
CAM_ERR(CAM_CSIPHY, "Invalid CSIPHY offset");
|
|
rc = -EINVAL;
|
|
} else if (scm_call2(SCM_SIP_FNID(0x18, 0x7), &description)) {
|
|
CAM_ERR(CAM_CSIPHY, "SCM call to hypervisor failed");
|
|
rc = -EINVAL;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
#endif
|
|
|
|
/* Callback to compare device from match list before adding as component */
|
|
static inline int camera_component_compare_dev(struct device *dev, void *data)
|
|
{
|
|
return dev == data;
|
|
}
|
|
|
|
/* Add component matches to list for master of aggregate driver */
|
|
int camera_component_match_add_drivers(struct device *master_dev,
|
|
struct component_match **match_list)
|
|
{
|
|
int i, rc = 0;
|
|
struct platform_device *pdev = NULL;
|
|
struct i2c_client *client = NULL;
|
|
struct device *start_dev = NULL, *match_dev = NULL;
|
|
|
|
if (!master_dev || !match_list) {
|
|
CAM_ERR(CAM_UTIL, "Invalid parameters for component match add");
|
|
rc = -EINVAL;
|
|
goto end;
|
|
}
|
|
|
|
for (i = 0; i < ARRAY_SIZE(cam_component_platform_drivers); i++) {
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 4, 0)
|
|
struct device_driver const *drv =
|
|
&cam_component_platform_drivers[i]->driver;
|
|
const void *drv_ptr = (const void *)drv;
|
|
#else
|
|
struct device_driver *drv = &cam_component_platform_drivers[i]->driver;
|
|
void *drv_ptr = (void *)drv;
|
|
#endif
|
|
start_dev = NULL;
|
|
while ((match_dev = bus_find_device(&platform_bus_type,
|
|
start_dev, drv_ptr, &camera_platform_compare_dev))) {
|
|
put_device(start_dev);
|
|
pdev = to_platform_device(match_dev);
|
|
CAM_DBG(CAM_UTIL, "Adding matched component:%s", pdev->name);
|
|
component_match_add(master_dev, match_list,
|
|
camera_component_compare_dev, match_dev);
|
|
start_dev = match_dev;
|
|
}
|
|
put_device(start_dev);
|
|
}
|
|
|
|
for (i = 0; i < ARRAY_SIZE(cam_component_i2c_drivers); i++) {
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 4, 0)
|
|
struct device_driver const *drv =
|
|
&cam_component_i2c_drivers[i]->driver;
|
|
const void *drv_ptr = (const void *)drv;
|
|
#else
|
|
struct device_driver *drv = &cam_component_i2c_drivers[i]->driver;
|
|
void *drv_ptr = (void *)drv;
|
|
#endif
|
|
start_dev = NULL;
|
|
while ((match_dev = bus_find_device(&i2c_bus_type,
|
|
start_dev, drv_ptr, &camera_i2c_compare_dev))) {
|
|
put_device(start_dev);
|
|
client = to_i2c_client(match_dev);
|
|
CAM_DBG(CAM_UTIL, "Adding matched component:%s", client->name);
|
|
component_match_add(master_dev, match_list,
|
|
camera_component_compare_dev, match_dev);
|
|
start_dev = match_dev;
|
|
}
|
|
put_device(start_dev);
|
|
}
|
|
|
|
end:
|
|
return rc;
|
|
}
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 10, 0)
|
|
#include <linux/qcom-iommu-util.h>
|
|
void cam_check_iommu_faults(struct iommu_domain *domain,
|
|
struct cam_smmu_pf_info *pf_info)
|
|
{
|
|
struct qcom_iommu_fault_ids fault_ids = {0, 0, 0};
|
|
|
|
if (qcom_iommu_get_fault_ids(domain, &fault_ids))
|
|
CAM_ERR(CAM_SMMU, "Cannot get smmu fault ids");
|
|
else
|
|
CAM_ERR(CAM_SMMU, "smmu fault ids bid:%d pid:%d mid:%d",
|
|
fault_ids.bid, fault_ids.pid, fault_ids.mid);
|
|
|
|
pf_info->bid = fault_ids.bid;
|
|
pf_info->pid = fault_ids.pid;
|
|
pf_info->mid = fault_ids.mid;
|
|
}
|
|
#else
|
|
void cam_check_iommu_faults(struct iommu_domain *domain,
|
|
struct cam_smmu_pf_info *pf_info)
|
|
{
|
|
struct iommu_fault_ids fault_ids = {0, 0, 0};
|
|
|
|
if (iommu_get_fault_ids(domain, &fault_ids))
|
|
CAM_ERR(CAM_SMMU, "Error: Can not get smmu fault ids");
|
|
|
|
CAM_ERR(CAM_SMMU, "smmu fault ids bid:%d pid:%d mid:%d",
|
|
fault_ids.bid, fault_ids.pid, fault_ids.mid);
|
|
|
|
pf_info->bid = fault_ids.bid;
|
|
pf_info->pid = fault_ids.pid;
|
|
pf_info->mid = fault_ids.mid;
|
|
}
|
|
#endif
|
|
|
|
static int inline cam_subdev_list_cmp(struct cam_subdev *entry_1, struct cam_subdev *entry_2)
|
|
{
|
|
if (entry_1->close_seq_prior > entry_2->close_seq_prior)
|
|
return 1;
|
|
else if (entry_1->close_seq_prior < entry_2->close_seq_prior)
|
|
return -1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
#if (KERNEL_VERSION(5, 18, 0) <= LINUX_VERSION_CODE)
|
|
int cam_compat_util_get_dmabuf_va(struct dma_buf *dmabuf, uintptr_t *vaddr)
|
|
{
|
|
struct iosys_map mapping;
|
|
int error_code = dma_buf_vmap(dmabuf, &mapping);
|
|
|
|
if (error_code) {
|
|
*vaddr = 0;
|
|
} else {
|
|
*vaddr = (mapping.is_iomem) ?
|
|
(uintptr_t)mapping.vaddr_iomem : (uintptr_t)mapping.vaddr;
|
|
CAM_DBG(CAM_MEM,
|
|
"dmabuf=%p, *vaddr=%p, is_iomem=%d, vaddr_iomem=%p, vaddr=%p",
|
|
dmabuf, *vaddr, mapping.is_iomem, mapping.vaddr_iomem, mapping.vaddr);
|
|
}
|
|
|
|
return error_code;
|
|
}
|
|
|
|
void cam_compat_util_put_dmabuf_va(struct dma_buf *dmabuf, void *vaddr)
|
|
{
|
|
struct iosys_map mapping = IOSYS_MAP_INIT_VADDR(vaddr);
|
|
|
|
dma_buf_vunmap(dmabuf, &mapping);
|
|
}
|
|
|
|
#elif (KERNEL_VERSION(5, 15, 0) <= LINUX_VERSION_CODE)
|
|
int cam_compat_util_get_dmabuf_va(struct dma_buf *dmabuf, uintptr_t *vaddr)
|
|
{
|
|
struct dma_buf_map mapping;
|
|
int error_code = dma_buf_vmap(dmabuf, &mapping);
|
|
|
|
if (error_code) {
|
|
*vaddr = 0;
|
|
} else {
|
|
*vaddr = (mapping.is_iomem) ?
|
|
(uintptr_t)mapping.vaddr_iomem : (uintptr_t)mapping.vaddr;
|
|
CAM_DBG(CAM_MEM,
|
|
"dmabuf=%p, *vaddr=%p, is_iomem=%d, vaddr_iomem=%p, vaddr=%p",
|
|
dmabuf, *vaddr, mapping.is_iomem, mapping.vaddr_iomem, mapping.vaddr);
|
|
}
|
|
|
|
return error_code;
|
|
}
|
|
|
|
void cam_compat_util_put_dmabuf_va(struct dma_buf *dmabuf, void *vaddr)
|
|
{
|
|
struct dma_buf_map mapping = DMA_BUF_MAP_INIT_VADDR(vaddr);
|
|
|
|
dma_buf_vunmap(dmabuf, &mapping);
|
|
}
|
|
|
|
#else
|
|
int cam_compat_util_get_dmabuf_va(struct dma_buf *dmabuf, uintptr_t *vaddr)
|
|
{
|
|
int error_code = 0;
|
|
void *addr = dma_buf_vmap(dmabuf);
|
|
|
|
if (!addr) {
|
|
*vaddr = 0;
|
|
error_code = -ENOSPC;
|
|
} else {
|
|
*vaddr = (uintptr_t)addr;
|
|
}
|
|
|
|
return error_code;
|
|
}
|
|
|
|
void cam_compat_util_put_dmabuf_va(struct dma_buf *dmabuf, void *vaddr)
|
|
{
|
|
dma_buf_vunmap(dmabuf, vaddr);
|
|
}
|
|
#endif
|
|
|
|
#if (KERNEL_VERSION(5, 15, 0) <= LINUX_VERSION_CODE)
|
|
void cam_smmu_util_iommu_custom(struct device *dev,
|
|
dma_addr_t discard_start, size_t discard_length)
|
|
{
|
|
|
|
}
|
|
|
|
int cam_req_mgr_ordered_list_cmp(void *priv,
|
|
const struct list_head *head_1, const struct list_head *head_2)
|
|
{
|
|
return cam_subdev_list_cmp(list_entry(head_1, struct cam_subdev, list),
|
|
list_entry(head_2, struct cam_subdev, list));
|
|
}
|
|
|
|
void cam_i3c_driver_remove(struct i3c_device *client)
|
|
{
|
|
CAM_DBG(CAM_SENSOR, "I3C remove invoked for %s",
|
|
(client ? dev_name(&client->dev) : "none"));
|
|
}
|
|
|
|
#else
|
|
void cam_smmu_util_iommu_custom(struct device *dev,
|
|
dma_addr_t discard_start, size_t discard_length)
|
|
{
|
|
iommu_dma_enable_best_fit_algo(dev);
|
|
|
|
if (discard_start)
|
|
iommu_dma_reserve_iova(dev, discard_start, discard_length);
|
|
|
|
return;
|
|
}
|
|
|
|
int cam_req_mgr_ordered_list_cmp(void *priv,
|
|
struct list_head *head_1, struct list_head *head_2)
|
|
{
|
|
return cam_subdev_list_cmp(list_entry(head_1, struct cam_subdev, list),
|
|
list_entry(head_2, struct cam_subdev, list));
|
|
}
|
|
|
|
int cam_i3c_driver_remove(struct i3c_device *client)
|
|
{
|
|
CAM_DBG(CAM_SENSOR, "I3C remove invoked for %s",
|
|
(client ? dev_name(&client->dev) : "none"));
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#if (KERNEL_VERSION(5, 15, 0) <= LINUX_VERSION_CODE && \
|
|
KERNEL_VERSION(5, 18, 0) > LINUX_VERSION_CODE)
|
|
long cam_dma_buf_set_name(struct dma_buf *dmabuf, const char *name)
|
|
{
|
|
long ret = 0;
|
|
|
|
ret = dma_buf_set_name(dmabuf, name);
|
|
|
|
return ret;
|
|
}
|
|
#else
|
|
long cam_dma_buf_set_name(struct dma_buf *dmabuf, const char *name)
|
|
{
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#if KERNEL_VERSION(5, 18, 0) <= LINUX_VERSION_CODE
|
|
void cam_eeprom_spi_driver_remove(struct spi_device *sdev)
|
|
{
|
|
struct v4l2_subdev *sd = spi_get_drvdata(sdev);
|
|
struct cam_eeprom_ctrl_t *e_ctrl;
|
|
struct cam_eeprom_soc_private *soc_private;
|
|
struct cam_hw_soc_info *soc_info;
|
|
|
|
if (!sd) {
|
|
CAM_ERR(CAM_EEPROM, "Subdevice is NULL");
|
|
return;
|
|
}
|
|
|
|
e_ctrl = (struct cam_eeprom_ctrl_t *)v4l2_get_subdevdata(sd);
|
|
if (!e_ctrl) {
|
|
CAM_ERR(CAM_EEPROM, "eeprom device is NULL");
|
|
return;
|
|
}
|
|
|
|
soc_info = &e_ctrl->soc_info;
|
|
mutex_lock(&(e_ctrl->eeprom_mutex));
|
|
cam_eeprom_shutdown(e_ctrl);
|
|
mutex_unlock(&(e_ctrl->eeprom_mutex));
|
|
mutex_destroy(&(e_ctrl->eeprom_mutex));
|
|
cam_unregister_subdev(&(e_ctrl->v4l2_dev_str));
|
|
kfree(e_ctrl->io_master_info.spi_client);
|
|
e_ctrl->io_master_info.spi_client = NULL;
|
|
soc_private =
|
|
(struct cam_eeprom_soc_private *)e_ctrl->soc_info.soc_private;
|
|
if (soc_private) {
|
|
kfree(soc_private->power_info.gpio_num_info);
|
|
soc_private->power_info.gpio_num_info = NULL;
|
|
kfree(soc_private);
|
|
soc_private = NULL;
|
|
}
|
|
v4l2_set_subdevdata(&e_ctrl->v4l2_dev_str.sd, NULL);
|
|
kfree(e_ctrl);
|
|
}
|
|
|
|
int cam_compat_util_get_irq(struct cam_hw_soc_info *soc_info)
|
|
{
|
|
int rc = 0;
|
|
|
|
soc_info->irq_num = platform_get_irq(soc_info->pdev, 0);
|
|
if (soc_info->irq_num < 0) {
|
|
rc = soc_info->irq_num;
|
|
return rc;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
#else
|
|
int cam_eeprom_spi_driver_remove(struct spi_device *sdev)
|
|
{
|
|
struct v4l2_subdev *sd = spi_get_drvdata(sdev);
|
|
struct cam_eeprom_ctrl_t *e_ctrl;
|
|
struct cam_eeprom_soc_private *soc_private;
|
|
struct cam_hw_soc_info *soc_info;
|
|
|
|
if (!sd) {
|
|
CAM_ERR(CAM_EEPROM, "Subdevice is NULL");
|
|
return -EINVAL;
|
|
}
|
|
|
|
e_ctrl = (struct cam_eeprom_ctrl_t *)v4l2_get_subdevdata(sd);
|
|
if (!e_ctrl) {
|
|
CAM_ERR(CAM_EEPROM, "eeprom device is NULL");
|
|
return -EINVAL;
|
|
}
|
|
|
|
soc_info = &e_ctrl->soc_info;
|
|
mutex_lock(&(e_ctrl->eeprom_mutex));
|
|
cam_eeprom_shutdown(e_ctrl);
|
|
mutex_unlock(&(e_ctrl->eeprom_mutex));
|
|
mutex_destroy(&(e_ctrl->eeprom_mutex));
|
|
cam_unregister_subdev(&(e_ctrl->v4l2_dev_str));
|
|
kfree(e_ctrl->io_master_info.spi_client);
|
|
e_ctrl->io_master_info.spi_client = NULL;
|
|
soc_private =
|
|
(struct cam_eeprom_soc_private *)e_ctrl->soc_info.soc_private;
|
|
if (soc_private) {
|
|
kfree(soc_private->power_info.gpio_num_info);
|
|
soc_private->power_info.gpio_num_info = NULL;
|
|
kfree(soc_private);
|
|
soc_private = NULL;
|
|
}
|
|
v4l2_set_subdevdata(&e_ctrl->v4l2_dev_str.sd, NULL);
|
|
kfree(e_ctrl);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int cam_compat_util_get_irq(struct cam_hw_soc_info *soc_info)
|
|
{
|
|
int rc = 0;
|
|
|
|
soc_info->irq_line =
|
|
platform_get_resource_byname(soc_info->pdev,
|
|
IORESOURCE_IRQ, soc_info->irq_name);
|
|
if (!soc_info->irq_line) {
|
|
rc = -ENODEV;
|
|
return rc;
|
|
}
|
|
soc_info->irq_num = soc_info->irq_line->start;
|
|
|
|
return rc;
|
|
}
|
|
#endif
|