2023-08-30 16:21:18 +08:00
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2022-2023, ArtInChip Technology Co., Ltd
|
|
|
|
|
*
|
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
*
|
|
|
|
|
* Authors: Siyao.Li <lisy@artinchip.com>
|
|
|
|
|
*/
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <getopt.h>
|
|
|
|
|
#include <sys/time.h>
|
|
|
|
|
#include <rtthread.h>
|
|
|
|
|
|
2023-11-09 20:19:51 +08:00
|
|
|
#include "hal_adcim.h"
|
2023-08-30 16:21:18 +08:00
|
|
|
#include "rtdevice.h"
|
|
|
|
|
#include "aic_core.h"
|
|
|
|
|
#include "aic_log.h"
|
2024-06-04 19:00:30 +08:00
|
|
|
#include "hal_gpai.h"
|
2023-08-30 16:21:18 +08:00
|
|
|
|
|
|
|
|
/* Global macro and variables */
|
|
|
|
|
#define AIC_GPAI_NAME "gpai"
|
2024-04-03 16:40:57 +08:00
|
|
|
/* The default voltages are set to D21x->3.0V, D31x、D12x->2.5V */
|
2023-08-30 16:21:18 +08:00
|
|
|
#define AIC_GPAI_DEFAULT_VOLTAGE 3
|
2024-01-27 08:47:24 +08:00
|
|
|
#define AIC_GPAI_VOLTAGE_ACCURACY 10000
|
2024-06-04 19:00:30 +08:00
|
|
|
#define AIC_GPAI_FIFO_MAX_DEPTH 32
|
2023-08-30 16:21:18 +08:00
|
|
|
|
|
|
|
|
static rt_adc_device_t gpai_dev;
|
2024-04-03 16:40:57 +08:00
|
|
|
static const char sopts[] = "c:t:n:h";
|
2023-08-30 16:21:18 +08:00
|
|
|
static const struct option lopts[] = {
|
|
|
|
|
{"channel", required_argument, NULL, 'c'},
|
|
|
|
|
{"voltage", required_argument, NULL, 't'},
|
2024-04-03 16:40:57 +08:00
|
|
|
{"number", required_argument, NULL, 'n'},
|
2023-08-30 16:21:18 +08:00
|
|
|
{"help", no_argument, NULL, 'h'},
|
|
|
|
|
{0, 0, 0, 0}
|
|
|
|
|
};
|
|
|
|
|
|
2024-04-03 16:40:57 +08:00
|
|
|
static int g_sample_num = -1;
|
2024-06-04 19:00:30 +08:00
|
|
|
static u32 g_cal_param;
|
|
|
|
|
static rt_sem_t g_gpai_sem = RT_NULL;
|
2023-08-30 16:21:18 +08:00
|
|
|
/* Functions */
|
|
|
|
|
|
|
|
|
|
static void cmd_gpai_usage(char *program)
|
|
|
|
|
{
|
|
|
|
|
printf("Compile time: %s %s\n", __DATE__, __TIME__);
|
|
|
|
|
printf("Usage: %s [options]\n", program);
|
2023-11-09 20:19:51 +08:00
|
|
|
printf("\t -c, --channel\t\tSelect one channel in [0, %d], default is 0\n",
|
2024-01-27 08:47:24 +08:00
|
|
|
AIC_GPAI_CH_NUM - 1);
|
|
|
|
|
printf("\t -t, --voltage\t\tModify default voltage\n");
|
2024-04-03 16:40:57 +08:00
|
|
|
printf("\t -n, --number\t\tSet the number of samples\n");
|
2023-08-30 16:21:18 +08:00
|
|
|
printf("\t -h, --help \n");
|
|
|
|
|
printf("\n");
|
2024-04-03 16:40:57 +08:00
|
|
|
printf("Example: %s -c 4 -n 100 -t 3\n", program);
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
|
|
|
|
|
2024-06-04 19:00:30 +08:00
|
|
|
static void gpai_irq_callback(void *arg)
|
|
|
|
|
{
|
|
|
|
|
rt_sem_release(g_gpai_sem);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int gpai_get_adc_period(struct aic_gpai_ch_info ch_info,
|
|
|
|
|
float def_voltage)
|
|
|
|
|
{
|
|
|
|
|
int cnt = 0;
|
|
|
|
|
int voltage = 0;
|
|
|
|
|
rt_base_t level;
|
|
|
|
|
int scale = AIC_GPAI_VOLTAGE_ACCURACY;
|
|
|
|
|
int sample_cnt = g_sample_num;
|
|
|
|
|
struct aic_gpai_irq_info irq_info;
|
|
|
|
|
int ch = ch_info.chan_id;
|
|
|
|
|
|
|
|
|
|
g_cal_param = hal_adcim_auto_calibration();
|
|
|
|
|
|
|
|
|
|
irq_info.chan_id = ch;
|
|
|
|
|
irq_info.callback = gpai_irq_callback;
|
|
|
|
|
irq_info.callback_param = NULL;
|
|
|
|
|
|
|
|
|
|
if (!g_gpai_sem)
|
|
|
|
|
g_gpai_sem = rt_sem_create("gpai_period_sem", 0, RT_IPC_FLAG_FIFO);
|
|
|
|
|
|
|
|
|
|
rt_adc_control(gpai_dev, RT_ADC_CMD_IRQ_CALLBACK, &irq_info);
|
|
|
|
|
rt_adc_enable(gpai_dev, ch);
|
|
|
|
|
|
|
|
|
|
while (cnt < sample_cnt) {
|
|
|
|
|
if (rt_sem_take(g_gpai_sem, RT_WAITING_FOREVER)!=RT_EOK)
|
|
|
|
|
break;
|
|
|
|
|
level = rt_hw_interrupt_disable();
|
|
|
|
|
rt_adc_control(gpai_dev, RT_ADC_CMD_GET_CH_INFO, (void *)&ch_info);
|
|
|
|
|
rt_hw_interrupt_enable(level);
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < ch_info.fifo_valid_cnt; i++) {
|
|
|
|
|
rt_kprintf("[%d] GPAI ch%d: %d\n", cnt, ch, ch_info.adc_values[i]);
|
|
|
|
|
if (ch_info.adc_values[i]) {
|
|
|
|
|
voltage = hal_adcim_adc2voltage(ch_info.adc_values[i],
|
|
|
|
|
g_cal_param,
|
|
|
|
|
AIC_GPAI_VOLTAGE_ACCURACY,
|
|
|
|
|
def_voltage);
|
|
|
|
|
rt_kprintf("GPAI voltage:%d.%04d v\n", voltage / scale,
|
|
|
|
|
voltage % scale);
|
|
|
|
|
}
|
|
|
|
|
cnt++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rt_adc_disable(gpai_dev, ch);
|
|
|
|
|
|
|
|
|
|
return voltage;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int gpai_get_adc_single(struct aic_gpai_ch_info ch_info,
|
|
|
|
|
float def_voltage)
|
2023-08-30 16:21:18 +08:00
|
|
|
{
|
2024-04-03 16:40:57 +08:00
|
|
|
int cnt = 0;
|
|
|
|
|
int voltage = 0;
|
|
|
|
|
int scale = AIC_GPAI_VOLTAGE_ACCURACY;
|
|
|
|
|
int sample_cnt = g_sample_num;
|
2024-06-04 19:00:30 +08:00
|
|
|
int adc_val;
|
|
|
|
|
int ch = ch_info.chan_id;
|
2024-04-03 16:40:57 +08:00
|
|
|
|
2024-06-04 19:00:30 +08:00
|
|
|
g_cal_param = hal_adcim_auto_calibration();
|
2023-08-30 16:21:18 +08:00
|
|
|
gpai_dev = (rt_adc_device_t)rt_device_find(AIC_GPAI_NAME);
|
|
|
|
|
if (!gpai_dev) {
|
|
|
|
|
rt_kprintf("Failed to open %s device\n", AIC_GPAI_NAME);
|
2024-06-04 19:00:30 +08:00
|
|
|
return -RT_ERROR;
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
|
|
|
|
|
2024-06-04 19:00:30 +08:00
|
|
|
rt_adc_enable(gpai_dev, ch);
|
2024-04-03 16:40:57 +08:00
|
|
|
|
2024-06-04 19:00:30 +08:00
|
|
|
while (cnt < sample_cnt) {
|
|
|
|
|
rt_adc_control(gpai_dev, RT_ADC_CMD_GET_CH_INFO, (void *)&ch_info);
|
2024-04-03 16:40:57 +08:00
|
|
|
|
2024-06-04 19:00:30 +08:00
|
|
|
aicos_msleep(10);
|
|
|
|
|
adc_val = ch_info.adc_values[0];
|
|
|
|
|
rt_kprintf("[%d] GPAI ch%d: %d\n", cnt, ch, adc_val);
|
|
|
|
|
if (adc_val) {
|
|
|
|
|
voltage = hal_adcim_adc2voltage(adc_val, g_cal_param,
|
2024-04-03 16:40:57 +08:00
|
|
|
AIC_GPAI_VOLTAGE_ACCURACY,
|
|
|
|
|
def_voltage);
|
2024-06-04 19:00:30 +08:00
|
|
|
rt_kprintf("GPAI voltage:%d.%04d v\n", voltage / scale,
|
2024-04-03 16:40:57 +08:00
|
|
|
voltage % scale);
|
|
|
|
|
}
|
2024-06-04 19:00:30 +08:00
|
|
|
cnt++;
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
|
|
|
|
|
2024-06-04 19:00:30 +08:00
|
|
|
rt_adc_disable(gpai_dev, ch);
|
2023-08-30 16:21:18 +08:00
|
|
|
|
2024-04-03 16:40:57 +08:00
|
|
|
return voltage;
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
|
|
|
|
|
2024-04-03 16:40:57 +08:00
|
|
|
static int cmd_test_gpai(int argc, char **argv)
|
2023-08-30 16:21:18 +08:00
|
|
|
{
|
|
|
|
|
int c;
|
2024-06-04 19:00:30 +08:00
|
|
|
int ret;
|
2023-08-30 16:21:18 +08:00
|
|
|
u32 ch = 0;
|
2024-01-27 08:47:24 +08:00
|
|
|
float def_voltage = AIC_GPAI_DEFAULT_VOLTAGE;
|
2024-04-03 16:40:57 +08:00
|
|
|
int voltage = -1;
|
2024-06-04 19:00:30 +08:00
|
|
|
struct aic_gpai_ch_info ch_info = {0};
|
2023-08-30 16:21:18 +08:00
|
|
|
|
|
|
|
|
if (argc < 2) {
|
|
|
|
|
cmd_gpai_usage(argv[0]);
|
2024-04-03 16:40:57 +08:00
|
|
|
return voltage;
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
optind = 0;
|
|
|
|
|
while ((c = getopt_long(argc, argv, sopts, lopts, NULL)) != -1) {
|
|
|
|
|
switch (c) {
|
|
|
|
|
case 'c':
|
|
|
|
|
ch = atoi(optarg);
|
2024-04-03 16:40:57 +08:00
|
|
|
if ((ch < 0) || (ch >= AIC_GPAI_CH_NUM))
|
2023-08-30 16:21:18 +08:00
|
|
|
pr_err("Invalid channel No.%s\n", optarg);
|
|
|
|
|
break;
|
|
|
|
|
case 't':
|
2024-01-27 08:47:24 +08:00
|
|
|
def_voltage = atof(optarg);
|
2023-08-30 16:21:18 +08:00
|
|
|
break;
|
2024-04-03 16:40:57 +08:00
|
|
|
case 'n':
|
|
|
|
|
g_sample_num = atoi(optarg);
|
|
|
|
|
break;
|
2023-08-30 16:21:18 +08:00
|
|
|
case 'h':
|
|
|
|
|
default:
|
|
|
|
|
cmd_gpai_usage(argv[0]);
|
2024-04-03 16:40:57 +08:00
|
|
|
return voltage;
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-03 16:40:57 +08:00
|
|
|
if (ch < 0) {
|
2023-08-30 16:21:18 +08:00
|
|
|
rt_kprintf("Please select a channel first\n");
|
2024-04-03 16:40:57 +08:00
|
|
|
return voltage;
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
2024-01-27 08:47:24 +08:00
|
|
|
if (def_voltage < 0) {
|
|
|
|
|
rt_kprintf("Please input valid standard voltage\n");
|
2024-04-03 16:40:57 +08:00
|
|
|
return voltage;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (g_sample_num <= 0) {
|
|
|
|
|
rt_kprintf("Please set the number of samples\n");
|
|
|
|
|
return voltage;
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
2024-06-04 19:00:30 +08:00
|
|
|
|
|
|
|
|
gpai_dev = (rt_adc_device_t)rt_device_find(AIC_GPAI_NAME);
|
|
|
|
|
if (!gpai_dev) {
|
|
|
|
|
rt_kprintf("Failed to open %s device\n", AIC_GPAI_NAME);
|
|
|
|
|
return -RT_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ch_info.chan_id = ch;
|
|
|
|
|
|
|
|
|
|
ret = rt_adc_control(gpai_dev, RT_ADC_CMD_GET_MODE, (void *)&ch_info);
|
|
|
|
|
if (ret) {
|
|
|
|
|
rt_kprintf("Failed to get GPAI mode\n");
|
|
|
|
|
return -RT_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (ch_info.mode) {
|
|
|
|
|
case AIC_GPAI_MODE_SINGLE:
|
|
|
|
|
rt_kprintf("Starting gpai single mode\n");
|
|
|
|
|
voltage = gpai_get_adc_single(ch_info, def_voltage);
|
|
|
|
|
break;
|
|
|
|
|
case AIC_GPAI_MODE_PERIOD:
|
|
|
|
|
rt_kprintf("Starting gpai period mode\n");
|
|
|
|
|
voltage = gpai_get_adc_period(ch_info, def_voltage);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
rt_kprintf("Unknown mode: %#x\n");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-30 16:21:18 +08:00
|
|
|
|
2024-04-03 16:40:57 +08:00
|
|
|
return voltage;
|
2023-08-30 16:21:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MSH_CMD_EXPORT_ALIAS(cmd_test_gpai, test_gpai, gpai device sample);
|