mirror of
git://nv-tegra.nvidia.com/linux-nv-oot.git
synced 2025-12-24 10:11:26 +03:00
Upstream Linux commit 4c1e0a97351a ("firmware: tegra: bpmp: Use
iosys-map helpers") updated the Tegra IVC driver to use the iosys-map
helpers for Linux v6.2. Update the DCE driver accordingly to work with
the latest Tegra IVC driver.
Bug 3936429
Change-Id: Id04fa3cbc078ed9657ae779b6a1bbd088008175c
Signed-off-by: Jon Hunter <jonathanh@nvidia.com>
Reviewed-on: https://git-master.nvidia.com/r/c/linux-nvidia/+/2842560
(cherry picked from commit 545f887e90cffff1d79f6a4ff0af87ddfcbeb178)
Reviewed-on: https://git-master.nvidia.com/r/c/linux-nvidia/+/2801234
Reviewed-by: Mahesh Kumar <mahkumar@nvidia.com>
Reviewed-by: svc-mobile-coverity <svc-mobile-coverity@nvidia.com>
Reviewed-by: svc-mobile-cert <svc-mobile-cert@nvidia.com>
Reviewed-by: svc_kernel_abi <svc_kernel_abi@nvidia.com>
GVS: Gerrit_Virtual_Submit <buildbot_gerritrpt@nvidia.com>
198 lines
5.0 KiB
C
198 lines
5.0 KiB
C
/*
|
|
* Copyright (c) 2019-2023, NVIDIA CORPORATION. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*/
|
|
|
|
#ifndef DCE_IPC_H
|
|
#define DCE_IPC_H
|
|
|
|
#include <linux/version.h>
|
|
#include <dce-lock.h>
|
|
#include <soc/tegra/ivc.h>
|
|
#include <interface/dce-admin-cmds.h>
|
|
#include <interface/dce-core-interface-ipc-types.h>
|
|
#include <interface/dce-ipc-state.h>
|
|
#include <linux/platform/tegra/dce/dce-client-ipc.h>
|
|
|
|
#define DCE_IPC_CHANNEL_TYPE_ADMIN 0U
|
|
#define DCE_IPC_CHANNEL_TYPE_CPU_CLIENTS 1U
|
|
|
|
#define DCE_IPC_MAX_IVC_CHANNELS 4U
|
|
|
|
/**
|
|
* TODO : Move the DispRM max to a config file
|
|
*/
|
|
#define DCE_DISPRM_CMD_MAX_NFRAMES 1U
|
|
#define DCE_DISPRM_CMD_MAX_FSIZE 4096U
|
|
#define DCE_DISPRM_EVENT_NOTIFY_CMD_MAX_NFRAMES 4U
|
|
#define DCE_DISPRM_EVENT_NOTIFY_CMD_MAX_FSIZE 4096U
|
|
#define DCE_ADMIN_CMD_MAX_FSIZE 1024U
|
|
|
|
#define DCE_IPC_WAIT_TYPE_INVALID 0U
|
|
#define DCE_IPC_WAIT_TYPE_RPC 1U
|
|
|
|
#define DCE_IPC_CHANNEL_VALID BIT(0)
|
|
#define DCE_IPC_CHANNEL_INITIALIZED BIT(1)
|
|
#define DCE_IPC_CHANNEL_SYNCED BIT(2)
|
|
#define DCE_IPC_CHANNEL_MSG_HEADER BIT(15)
|
|
|
|
#define DCE_IPC_CH_KMD_TYPE_ADMIN 0U
|
|
#define DCE_IPC_CH_KMD_TYPE_RM 1U
|
|
#define DCE_IPC_CH_KMD_TYPE_HDCP 2U
|
|
#define DCE_IPC_CH_KMD_TYPE_RM_NOTIFY 3U
|
|
#define DCE_IPC_CH_KMD_TYPE_MAX 4U
|
|
/**
|
|
* struct dce_ipc_signal - Stores ivc channel details
|
|
*
|
|
* @d : Pointer to struct tegra_dce.
|
|
* @ibuff : Pointer to the input data buffer.
|
|
* @obuff : Pointer to the output data buffer.
|
|
* @d_ivc : Pointer to the ivc data structure.
|
|
*/
|
|
struct dce_ipc_mailbox {
|
|
u8 mb_type;
|
|
u32 mb_num;
|
|
};
|
|
|
|
/**
|
|
* TODO : Use linux doorbell driver
|
|
*/
|
|
struct dce_ipc_doorbell {
|
|
u32 db_num;
|
|
u32 db_bit;
|
|
};
|
|
|
|
struct dce_ipc_signal_instance {
|
|
u32 type;
|
|
u32 sema_num;
|
|
u32 sema_bit;
|
|
union {
|
|
struct dce_ipc_mailbox mbox;
|
|
struct dce_ipc_doorbell db;
|
|
} form;
|
|
struct dce_ipc_signal *signal;
|
|
struct dce_ipc_signal_instance *next;
|
|
};
|
|
|
|
typedef void (*dce_ipc_signal_notify)(struct tegra_dce *d,
|
|
struct dce_ipc_signal_instance *signal);
|
|
|
|
struct dce_ipc_signal {
|
|
struct dce_ipc_channel *ch;
|
|
dce_ipc_signal_notify notify;
|
|
struct dce_ipc_signal_instance to_d;
|
|
struct dce_ipc_signal_instance from_d;
|
|
};
|
|
|
|
int dce_ipc_signal_init(struct dce_ipc_channel *chan);
|
|
|
|
|
|
/**
|
|
* struct dce_ipc_region - Contains ivc region specific memory info.
|
|
*
|
|
* @size : total IVC region size.
|
|
* @tx : transmit region info.
|
|
* @rx : receive region info.
|
|
*/
|
|
struct dce_ipc_region {
|
|
u32 s_offset;
|
|
dma_addr_t iova;
|
|
unsigned long size;
|
|
void __iomem *base;
|
|
};
|
|
|
|
struct dce_ipc_queue_info {
|
|
u8 nframes;
|
|
u32 frame_sz;
|
|
dma_addr_t rx_iova;
|
|
dma_addr_t tx_iova;
|
|
};
|
|
|
|
/**
|
|
* struct dce_ipc_channel - Stores ivc channel details
|
|
*
|
|
* @d : Pointer to struct tegra_dce.
|
|
* @ibuff : Pointer to the input data buffer.
|
|
* @obuff : Pointer to the output data buffer.
|
|
* @d_ivc : Pointer to the ivc data structure.
|
|
*/
|
|
struct dce_ipc_channel {
|
|
u32 flags;
|
|
u32 w_type;
|
|
u32 ch_type;
|
|
u32 ipc_type;
|
|
#if (KERNEL_VERSION(6, 2, 0) <= LINUX_VERSION_CODE)
|
|
struct iosys_map ibuff;
|
|
struct iosys_map obuff;
|
|
#else
|
|
void *ibuff;
|
|
void *obuff;
|
|
#endif
|
|
struct tegra_ivc d_ivc;
|
|
struct tegra_dce *d;
|
|
struct dce_mutex lock;
|
|
struct dce_ipc_signal signal;
|
|
struct dce_ipc_queue_info q_info;
|
|
};
|
|
|
|
/**
|
|
* struct dce_ipc - Stores ipc data
|
|
*
|
|
* @region - Store data about ivc region in DRAM
|
|
* @ch - Array of pointers to store dce ivc channel info
|
|
*/
|
|
struct dce_ipc {
|
|
struct dce_ipc_region region;
|
|
struct dce_ipc_channel *ch[DCE_IPC_MAX_IVC_CHANNELS];
|
|
};
|
|
|
|
int dce_ipc_send_message(struct tegra_dce *d,
|
|
u32 ch_type, const void *data, size_t size);
|
|
|
|
int dce_ipc_read_message(struct tegra_dce *d,
|
|
u32 ch_type, void *data, size_t size);
|
|
|
|
int dce_ipc_send_message_sync(struct tegra_dce *d,
|
|
u32 ch_type, struct dce_ipc_message *msg);
|
|
|
|
int dce_ipc_get_channel_info(struct tegra_dce *d,
|
|
struct dce_ipc_queue_info *q_info, u32 ch_index);
|
|
|
|
void dce_ipc_free_region(struct tegra_dce *d);
|
|
|
|
int dce_ipc_allocate_region(struct tegra_dce *d);
|
|
|
|
struct tegra_dce *dce_ipc_get_dce_from_ch(u32 ch_type);
|
|
|
|
int dce_ipc_channel_init(struct tegra_dce *d, u32 ch_type);
|
|
|
|
void dce_ipc_channel_deinit(struct tegra_dce *d, u32 ch_type);
|
|
|
|
void dce_ipc_channel_reset(struct tegra_dce *d, u32 ch_type);
|
|
|
|
uint32_t dce_ipc_get_ipc_type(struct tegra_dce *d, u32 ch_type);
|
|
|
|
bool dce_ipc_channel_is_ready(struct tegra_dce *d, u32 ch_type);
|
|
|
|
bool dce_ipc_channel_is_synced(struct tegra_dce *d, u32 ch_type);
|
|
|
|
u32 dce_ipc_get_cur_wait_type(struct tegra_dce *d, u32 ch_type);
|
|
|
|
bool dce_ipc_is_data_available(struct tegra_dce *d, u32 ch_type);
|
|
|
|
int dce_ipc_get_region_iova_info(struct tegra_dce *d, u64 *iova, u32 *size);
|
|
|
|
int dce_ipc_init_signaling(struct tegra_dce *d, struct dce_ipc_channel *ch);
|
|
|
|
void dce_ipc_deinit_signaling(struct tegra_dce *d, struct dce_ipc_channel *ch);
|
|
|
|
#endif
|