/* * This file is part of the coreboot project. * * Copyright 2016 Rockchip Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; version 2 of the License. * * This program is distributed in the hope that 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. * */ #include <arch/io.h> #include <assert.h> #include <console/console.h> #include <delay.h> #include <soc/clock.h> #include <soc/saradc.h> #include <stdint.h> #include <stdlib.h> #include <timer.h> struct rk3399_saradc_regs { u32 data; u32 stas; u32 ctrl; u32 dly_pu_soc; }; check_member(rk3399_saradc_regs, dly_pu_soc, 0xc); struct rk3399_saradc_regs *rk3399_saradc = (void *)SARADC_BASE; /* SARADC_STAS: conversion done */ #define ADC_STOP 0 /* SARADC_CTRL */ #define INT_EN (1 << 5) #define ADC_PWR_CTRL (1 << 3) #define ADC_CHN_SEL_MASK 7 #define ADC_CHN_SEL_SHIFT 0 /* SARADC_DATA, 10[0:9] bits */ #define DATA_MASK 0x3FF /* The max clk is 13 MHz, we also recommended that * the sample rate(=clk/13) should be > 500KHz. * So choose 8MHz, that 8MHz/13 = 615.38KHz > 500KHz. */ #define SARADC_HZ (8*MHz) /* TRM(V0.3 Part 1 Page 366) said there is a delay between * power up and start command, default value is 2 src clk. * Let delay 2 src clk here, in ns(udelay). */ #define SARADC_DELAY_PU (1 * 1000 * 1000 * 1000 / SARADC_HZ * 2) #define SARADC_MAX_CHANNEL 6 u32 get_saradc_value(u32 chn) { u32 adc_value; struct stopwatch sw; assert(chn < SARADC_MAX_CHANNEL); rkclk_configure_saradc(SARADC_HZ); /* power down adc converter */ clrbits_le32(&rk3399_saradc->ctrl, ADC_PWR_CTRL); /* select channel */ clrsetbits_le32(&rk3399_saradc->ctrl, ADC_CHN_SEL_MASK << ADC_CHN_SEL_SHIFT, chn << ADC_CHN_SEL_SHIFT); /* power up */ setbits_le32(&rk3399_saradc->ctrl, ADC_PWR_CTRL); udelay(SARADC_DELAY_PU); stopwatch_init_msecs_expire(&sw, 10); do { if (read32(&rk3399_saradc->stas) == ADC_STOP) { adc_value = read32(&rk3399_saradc->data) & DATA_MASK; return adc_value; } } while (!stopwatch_expired(&sw)); return -1; }