kernel_samsung_a34x-permissive/drivers/misc/mediatek/eccci/hif/ccci_hif.c
2024-04-28 15:51:13 +02:00

395 lines
9.3 KiB
C

// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (C) 2016 MediaTek Inc.
*/
#include <linux/types.h>
#include <linux/ktime.h>
#include "ccci_debug.h"
#include "ccci_core.h"
#include "ccci_hif_cldma.h"
#include "ccci_hif_dpmaif.h"
#include "ccci_config.h"
#define TAG "hif"
void *ccci_hif[CCCI_HIF_NUM];
struct ccci_hif_ops *ccci_hif_op[CCCI_HIF_NUM];
int ccci_hif_dump_status(unsigned int hif_flag,
enum MODEM_DUMP_FLAG dump_flag,
void *buff, int length)
{
int ret = 0;
if (hif_flag & (1 << CLDMA_HIF_ID))
ret |= ccci_cldma_hif_dump_status(CLDMA_HIF_ID,
dump_flag, buff, length);
if (hif_flag & (1 << CCIF_HIF_ID) && ccci_hif[CCIF_HIF_ID])
ret |= ccci_hif_op[CCIF_HIF_ID]->dump_status(CCIF_HIF_ID,
dump_flag, buff, length);
if (hif_flag & (1 << DPMAIF_HIF_ID) && ccci_hif[DPMAIF_HIF_ID])
ret |= ccci_hif_op[DPMAIF_HIF_ID]->dump_status(DPMAIF_HIF_ID,
dump_flag, buff, length);
return ret;
}
int ccci_hif_debug(unsigned char hif_id, enum ccci_hif_debug_flg debug_id,
int *paras, int len)
{
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->debug)
return ccci_hif_op[hif_id]->debug(hif_id, debug_id, paras);
else
return 0;
}
void *ccci_hif_fill_rt_header(unsigned char hif_id,
int packet_size, unsigned int tx_ch, unsigned int txqno)
{
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->fill_rt_header)
return ccci_hif_op[hif_id]->fill_rt_header(hif_id,
packet_size, tx_ch, txqno);
CCCI_ERROR_LOG(-1, CORE, "rt header : %d\n", hif_id);
return NULL;
}
int ccci_hif_set_wakeup_src(unsigned char hif_id, int value)
{
int ret = 0;
switch (hif_id) {
case CLDMA_HIF_ID:
ret = ccci_cldma_hif_set_wakeup_src(CLDMA_HIF_ID, value);
break;
case CCIF_HIF_ID:
case DPMAIF_HIF_ID:
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->debug)
ret = ccci_hif_op[hif_id]->debug(hif_id,
CCCI_HIF_DEBUG_SET_WAKEUP, &value);
break;
default:
break;
}
return ret;
}
int ccci_hif_send_skb(unsigned char hif_id, int tx_qno, struct sk_buff *skb,
int from_pool, int blocking)
{
int ret = 0;
switch (hif_id) {
case CLDMA_HIF_ID:
ret = ccci_cldma_hif_send_skb(CLDMA_HIF_ID, tx_qno,
skb, from_pool, blocking);
break;
case CCIF_HIF_ID:
case DPMAIF_HIF_ID:
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->send_skb)
ret = ccci_hif_op[hif_id]->send_skb(hif_id,
tx_qno, skb, from_pool, blocking);
CCCI_HISTORY_TAG_LOG(-1, TAG,
"%s: %d (%p, %p)\n", __func__,
hif_id, ccci_hif[hif_id], ccci_hif_op[hif_id]);
break;
default:
break;
}
return ret;
}
int ccci_hif_send_data(unsigned char hif_id, int tx_qno)
{
int ret = 0;
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->send_data)
ret = ccci_hif_op[hif_id]->send_data(hif_id, tx_qno);
return ret;
}
int ccci_hif_write_room(unsigned char hif_id, unsigned char qno)
{
int ret = 0;
switch (hif_id) {
case CLDMA_HIF_ID:
ret = ccci_cldma_hif_write_room(CLDMA_HIF_ID, qno);
break;
case CCIF_HIF_ID:
case DPMAIF_HIF_ID:
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->write_room)
ret = ccci_hif_op[hif_id]->write_room(hif_id, qno);
break;
default:
break;
}
return ret;
}
int ccci_hif_ask_more_request(unsigned char hif_id, int rx_qno)
{
int ret = 0;
switch (hif_id) {
case CLDMA_HIF_ID:
ret = ccci_cldma_hif_give_more(CLDMA_HIF_ID, rx_qno);
break;
case CCIF_HIF_ID:
case DPMAIF_HIF_ID:
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->give_more)
ret = ccci_hif_op[hif_id]->give_more(hif_id, rx_qno);
break;
default:
break;
}
return ret;
}
void ccci_hif_start_queue(unsigned char hif_id, unsigned int reserved,
enum DIRECTION dir)
{
}
static inline int ccci_hif_napi_poll(unsigned char md_id, int rx_qno,
struct napi_struct *napi, int weight)
{
return 0;
}
static void ccci_md_dump_log_rec(unsigned char md_id, struct ccci_log *log)
{
u64 ts_nsec = log->tv;
unsigned long rem_nsec;
if (ts_nsec == 0)
return;
rem_nsec = do_div(ts_nsec, 1000000000);
if (!log->droped) {
CCCI_MEM_LOG(md_id, CORE,
"%08X %08X %08X %08X %5lu.%06lu\n",
log->msg.data[0], log->msg.data[1],
*(((u32 *)&log->msg) + 2),
log->msg.reserved, (unsigned long)ts_nsec, rem_nsec / 1000);
} else {
CCCI_MEM_LOG(md_id, CORE, "%08X %08X %08X %08X %5lu.%06lu -\n",
log->msg.data[0], log->msg.data[1],
*(((u32 *)&log->msg) + 2),
log->msg.reserved, (unsigned long)ts_nsec,
rem_nsec / 1000);
}
}
void ccci_md_add_log_history(struct ccci_hif_traffic *tinfo,
enum DIRECTION dir,
int queue_index, struct ccci_header *msg, int is_droped)
{
#ifdef PACKET_HISTORY_DEPTH
if (queue_index < 0 || queue_index >= MAX_TXQ_NUM
|| tinfo->rx_history_ptr[queue_index] < 0
|| tinfo->rx_history_ptr[queue_index] >= PACKET_HISTORY_DEPTH
|| tinfo->tx_history_ptr[queue_index] < 0
|| tinfo->tx_history_ptr[queue_index] >= PACKET_HISTORY_DEPTH) {
CCCI_MEM_LOG(-1, CORE,
"invalid queue_index=%d\n", queue_index);
return;
}
if (dir == OUT) {
memcpy(&tinfo->tx_history[queue_index][
tinfo->tx_history_ptr[queue_index]].msg, msg,
sizeof(struct ccci_header));
tinfo->tx_history[queue_index][
tinfo->tx_history_ptr[queue_index]].tv
= local_clock();
tinfo->tx_history[queue_index][
tinfo->tx_history_ptr[queue_index]].droped
= is_droped;
tinfo->tx_history_ptr[queue_index]++;
tinfo->tx_history_ptr[queue_index]
&= (unsigned int)(PACKET_HISTORY_DEPTH - 1);
}
if (dir == IN) {
memcpy(&tinfo->rx_history[queue_index][
tinfo->rx_history_ptr[queue_index]].msg, msg,
sizeof(struct ccci_header));
tinfo->rx_history[queue_index][
tinfo->rx_history_ptr[queue_index]].tv = local_clock();
tinfo->rx_history[queue_index][
tinfo->rx_history_ptr[queue_index]].droped = is_droped;
tinfo->rx_history_ptr[queue_index]++;
tinfo->rx_history_ptr[queue_index]
&= (PACKET_HISTORY_DEPTH - 1);
}
#endif
}
EXPORT_SYMBOL(ccci_md_add_log_history);
void ccci_md_dump_log_history(unsigned char md_id,
struct ccci_hif_traffic *tinfo, int dump_multi_rec,
int tx_queue_num, int rx_queue_num)
{
#ifdef PACKET_HISTORY_DEPTH
int i_tx, i_rx, j;
int tx_qno, rx_qno;
if (!dump_multi_rec && (tx_queue_num >= MAX_TXQ_NUM ||
rx_queue_num >= MAX_RXQ_NUM))
return;
if (dump_multi_rec) {
tx_qno = ((tx_queue_num <= MAX_TXQ_NUM) ?
tx_queue_num : MAX_TXQ_NUM);
rx_qno = ((rx_queue_num <= MAX_RXQ_NUM) ?
rx_queue_num : MAX_RXQ_NUM);
i_tx = 0;
i_rx = 0;
} else {
tx_qno = tx_queue_num + 1;
rx_qno = rx_queue_num + 1;
i_tx = tx_queue_num;
i_rx = rx_queue_num;
}
if (rx_queue_num > 0)
for (; i_rx < rx_qno; i_rx++) {
CCCI_MEM_LOG_TAG(md_id, CORE,
"dump rxq%d packet history, ptr=%d\n", i_rx,
tinfo->rx_history_ptr[i_rx]);
for (j = 0; j < PACKET_HISTORY_DEPTH; j++)
ccci_md_dump_log_rec(md_id,
&tinfo->rx_history[i_rx][j]);
}
if (tx_queue_num > 0)
for (; i_tx < tx_qno; i_tx++) {
CCCI_MEM_LOG_TAG(md_id, CORE,
"dump txq%d packet history, ptr=%d\n", i_tx,
tinfo->tx_history_ptr[i_tx]);
for (j = 0; j < PACKET_HISTORY_DEPTH; j++)
ccci_md_dump_log_rec(md_id,
&tinfo->tx_history[i_tx][j]);
}
#endif
}
EXPORT_SYMBOL(ccci_md_dump_log_history);
void ccci_hif_md_exception(unsigned int hif_flag, unsigned char stage)
{
switch (stage) {
case HIF_EX_INIT:
/* eg. stop tx */
break;
case HIF_EX_CLEARQ_DONE:
/* eg. stop rx. */
break;
case HIF_EX_ALLQ_RESET:
/* maybe no used for dpmaif, for no used on exception mode. */
break;
default:
break;
};
}
int ccci_hif_stop(unsigned char hif_id)
{
int ret = 0;
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->stop)
ret |= ccci_hif_op[hif_id]->stop(hif_id);
return ret;
}
int ccci_hif_start(unsigned char hif_id)
{
int ret = 0;
if (ccci_hif[hif_id] && ccci_hif_op[hif_id]->start)
ret |= ccci_hif_op[hif_id]->start(hif_id);
return ret;
}
int ccci_hif_state_notification(int md_id, unsigned char state)
{
int ret = 0;
switch (state) {
case BOOT_WAITING_FOR_HS1:
ccci_hif_start(CCIF_HIF_ID);
#if MD_GENERATION >= (6295)
ccci_hif_start(DPMAIF_HIF_ID);
#else
ccci_hif_start(CLDMA_HIF_ID);
#endif
break;
case READY:
break;
case RESET:
break;
case EXCEPTION:
case WAITING_TO_STOP:
if (ccci_hif[DPMAIF_HIF_ID] &&
ccci_hif_op[DPMAIF_HIF_ID]->pre_stop) {
ccci_hif_dump_status(1 << DPMAIF_HIF_ID,
DUMP_FLAG_REG, NULL, -1);
ccci_hif_op[DPMAIF_HIF_ID]->pre_stop(DPMAIF_HIF_ID);
}
break;
case GATED:
if (ccci_hif[CCIF_HIF_ID] && ccci_hif_op[CCIF_HIF_ID]->stop)
ret |= ccci_hif_op[CCIF_HIF_ID]->stop(CCIF_HIF_ID);
/* later than ccmni */
if (ccci_hif[DPMAIF_HIF_ID] &&
ccci_hif_op[DPMAIF_HIF_ID]->stop) {
ccci_hif_dump_status(1 << DPMAIF_HIF_ID,
DUMP_FLAG_REG, NULL, -1);
ret |= ccci_hif_op[DPMAIF_HIF_ID]->stop(DPMAIF_HIF_ID);
}
break;
default:
break;
}
return ret;
}
void ccci_hif_resume(unsigned char md_id, unsigned int hif_flag)
{
}
void ccci_hif_suspend(unsigned char md_id, unsigned int hif_flag)
{
}
void ccci_hif_register(unsigned char hif_id, void *hif_per_data,
struct ccci_hif_ops *ops)
{
CCCI_NORMAL_LOG(0, CORE, "hif register: %d\n", hif_id);
CCCI_HISTORY_TAG_LOG(0, CORE,
"hif register: %d\n", hif_id);
if (hif_id < CCCI_HIF_NUM) {
ccci_hif[hif_id] = hif_per_data;
ccci_hif_op[hif_id] = ops;
}
}
EXPORT_SYMBOL(ccci_hif_register);
#ifdef CCCI_KMODULE_ENABLE
void *ccci_hif_get_by_id(unsigned char hif_id)
{
if (hif_id >= CCCI_HIF_NUM) {
CCCI_ERROR_LOG(-1, CORE,
"%s hif_id = %u\n", __func__, hif_id);
return NULL;
} else
return ccci_hif[hif_id];
}
EXPORT_SYMBOL(ccci_hif_get_by_id);
#endif