Files
OpenCellular/chip/npcx/adc.c
Vijay Hiremath e7bebf7c80 ADC: Add common priority for the ADC init
Added common priority for the ADC init to ensure board level
priority is defined for the ADC pre init.

BUG=none
TEST=make buildall -j
BRANCH=none

Change-Id: Id1649df6a68ab53bd110e58a0722bd4c70cbffc5
Signed-off-by: Vijay Hiremath <vijay.p.hiremath@intel.com>
Reviewed-on: https://chromium-review.googlesource.com/286040
Reviewed-by: Shawn N <shawnn@chromium.org>
2015-07-16 05:15:22 +00:00

239 lines
5.8 KiB
C

/* Copyright (c) 2014 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
/* NPCX-specific ADC module for Chrome EC */
#include "adc.h"
#include "adc_chip.h"
#include "atomic.h"
#include "clock.h"
#include "clock_chip.h"
#include "console.h"
#include "common.h"
#include "gpio.h"
#include "hooks.h"
#include "registers.h"
#include "task.h"
#include "timer.h"
#include "util.h"
/* Maximum time we allow for an ADC conversion */
#define ADC_TIMEOUT_US SECOND
#define ADC_CLK 2000000
#define ADC_REGULAR_DLY 0x11
#define ADC_REGULAR_ADCCNF2 0x8B07
#define ADC_REGULAR_GENDLY 0x0100
#define ADC_REGULAR_MEAST 0x0001
/* ADC conversion mode */
enum npcx_adc_conversion_mode {
ADC_CHN_CONVERSION_MODE = 0,
ADC_SCAN_CONVERSION_MODE = 1
};
/* ADC repetitive mode */
enum npcx_adc_repetitive_mode {
ADC_ONE_SHOT_CONVERSION_TYPE = 0,
ADC_REPETITIVE_CONVERSION_TYPE = 1
};
/* Global variables */
static task_id_t task_waiting;
/**
* Preset ADC operation clock.
*
* @param none
* @return none
* @notes changed when initial or HOOK_FREQ_CHANGE command
*/
void adc_freq_changed(void)
{
uint8_t prescaler_divider = 0;
/* Set clock prescaler divider to ADC module*/
prescaler_divider = (uint8_t)(clock_get_apb1_freq() / ADC_CLK);
if (prescaler_divider >= 1)
prescaler_divider = prescaler_divider - 1;
if (prescaler_divider > 0x3F)
prescaler_divider = 0x3F;
/* Set Core Clock Division Factor in order to obtain the ADC clock */
NPCX_ATCTL = (NPCX_ATCTL & (~(((1<<6)-1)<<NPCX_ATCTL_SCLKDIV)))
|(prescaler_divider<<NPCX_ATCTL_SCLKDIV);
}
DECLARE_HOOK(HOOK_FREQ_CHANGE, adc_freq_changed, HOOK_PRIO_DEFAULT);
/**
* Get current voltage data of the specified channel.
*
* @param input_ch npcx input channel to read
* @return ADC channel voltage data.(Range: 0~1023)
*/
static int get_channel_data(enum npcx_adc_input_channel input_ch)
{
return (NPCX_CHNDAT(input_ch)>>NPCX_CHNDAT_CHDAT) & ((1<<10)-1);
}
/**
* Flush an ADC sequencer and initiate a read.
*
* @param input_ch operation channel
* @param timeout preset timeout
* @return TRUE/FALSE success/fail
* @notes set SW-triggered interrupt conversion and one-shot mode in npcx chip
*/
static int start_single_and_wait(enum npcx_adc_input_channel input_ch
, int timeout)
{
int event;
task_waiting = task_get_current();
/* Set ADC conversion code to SW conversion mode */
NPCX_ADCCNF = (NPCX_ADCCNF & (~(((1<<2)-1)<<NPCX_ADCCNF_ADCMD)))
|(ADC_CHN_CONVERSION_MODE<<NPCX_ADCCNF_ADCMD);
/* Set conversion type to one-shot type */
NPCX_ADCCNF = (NPCX_ADCCNF & (~(((1<<1)-1)<<NPCX_ADCCNF_ADCRPTC)))
|(ADC_ONE_SHOT_CONVERSION_TYPE<<NPCX_ADCCNF_ADCRPTC);
/* Update number of channel to be converted */
NPCX_ASCADD = (NPCX_ASCADD & (~(((1<<5)-1)<<NPCX_ASCADD_SADDR)))
|(input_ch<<NPCX_ASCADD_SADDR);
/* Clear End-of-Conversion Event status */
SET_BIT(NPCX_ADCSTS, NPCX_ADCSTS_EOCEV);
/* Enable ADC End-of-Conversion Interrupt if applicable */
SET_BIT(NPCX_ADCCNF, NPCX_ADCCNF_INTECEN);
/* Start conversion */
SET_BIT(NPCX_ADCCNF, NPCX_ADCCNF_START);
/* Wait for interrupt */
event = task_wait_event(timeout);
task_waiting = TASK_ID_INVALID;
return event != TASK_EVENT_TIMER;
}
/**
* ADC read specific channel.
*
* @param ch operation channel
* @return ADC converted voltage or error message
*/
int adc_read_channel(enum adc_channel ch)
{
const struct adc_t *adc = adc_channels + ch;
static struct mutex adc_lock;
int value;
mutex_lock(&adc_lock);
if (start_single_and_wait(adc->input_ch, ADC_TIMEOUT_US)) {
if ((adc->input_ch ==
((NPCX_ASCADD>>NPCX_ASCADD_SADDR)&((1<<5)-1)))
&& (IS_BIT_SET(NPCX_CHNDAT(adc->input_ch),
NPCX_CHNDAT_NEW))) {
value = get_channel_data(adc->input_ch) *
adc->factor_mul / adc->factor_div + adc->shift;
} else {
value = ADC_READ_ERROR;
}
} else {
value = ADC_READ_ERROR;
}
mutex_unlock(&adc_lock);
return value;
}
/**
* ADC read all channels.
*
* @param data all ADC converted voltage
* @return ADC converted error message
*/
int adc_read_all_channels(int *data)
{
int i;
for (i = 0; i < ADC_CH_COUNT; ++i) {
data[i] = adc_read_channel(i);
if (ADC_READ_ERROR == data[i])
return EC_ERROR_UNKNOWN;
}
return EC_SUCCESS;
}
/**
* ADC interrupt handler
*
* @param none
* @return none
* @notes Only handle SW-triggered conversion in npcx chip
*/
void adc_interrupt(void)
{
if (IS_BIT_SET(NPCX_ADCSTS, NPCX_ADCSTS_EOCEV)) {
/* Disable End-of-Conversion Interrupt */
CLEAR_BIT(NPCX_ADCCNF, NPCX_ADCCNF_INTECEN);
/* Stop conversion */
SET_BIT(NPCX_ADCCNF, NPCX_ADCCNF_STOP);
/* Clear End-of-Conversion Event status */
SET_BIT(NPCX_ADCSTS, NPCX_ADCSTS_EOCEV);
/* Wake up the task which was waiting for the interrupt */
if (task_waiting != TASK_ID_INVALID)
task_wake(task_waiting);
}
}
DECLARE_IRQ(NPCX_IRQ_ADC, adc_interrupt, 2);
/**
* ADC initial.
*
* @param none
* @return none
*/
static void adc_init(void)
{
/* Configure pins from GPIOs to ADCs */
gpio_config_module(MODULE_ADC, 1);
/* Enable ADC clock (bit4 mask = 0x10) */
clock_enable_peripheral(CGC_OFFSET_ADC, CGC_ADC_MASK,
CGC_MODE_RUN | CGC_MODE_SLEEP);
/* Enable ADC */
SET_BIT(NPCX_ADCCNF, NPCX_ADCCNF_ADCEN);
/* Set Core Clock Division Factor in order to obtain the ADC clock */
adc_freq_changed();
/* Set regular speed */
NPCX_ATCTL = (NPCX_ATCTL & (~(((1<<3)-1)<<NPCX_ATCTL_DLY)))
|((ADC_REGULAR_DLY - 1)<<NPCX_ATCTL_DLY);
NPCX_ADCCNF2 = ADC_REGULAR_ADCCNF2;
NPCX_GENDLY = ADC_REGULAR_GENDLY;
NPCX_MEAST = ADC_REGULAR_MEAST;
task_waiting = TASK_ID_INVALID;
/* Enable IRQs */
task_enable_irq(NPCX_IRQ_ADC);
}
DECLARE_HOOK(HOOK_INIT, adc_init, HOOK_PRIO_INIT_ADC);