diff options
-rw-r--r-- | board/mec1322_evb/board.c | 11 | ||||
-rw-r--r-- | board/mec1322_evb/board.h | 10 | ||||
-rw-r--r-- | chip/mec1322/adc.c | 107 | ||||
-rw-r--r-- | chip/mec1322/adc_chip.h | 33 | ||||
-rw-r--r-- | chip/mec1322/build.mk | 1 | ||||
-rw-r--r-- | chip/mec1322/registers.h | 10 |
6 files changed, 172 insertions, 0 deletions
diff --git a/board/mec1322_evb/board.c b/board/mec1322_evb/board.c index 6a26913ca8..5d7dd42bb9 100644 --- a/board/mec1322_evb/board.c +++ b/board/mec1322_evb/board.c @@ -4,6 +4,8 @@ */ /* MEC1322 eval board-specific configuration */ +#include "adc.h" +#include "adc_chip.h" #include "fan.h" #include "gpio.h" #include "i2c.h" @@ -44,6 +46,15 @@ const struct gpio_alt_func gpio_alt_funcs[] = { }; const int gpio_alt_funcs_count = ARRAY_SIZE(gpio_alt_funcs); +/* ADC channels */ +const struct adc_t adc_channels[] = { + [ADC_CH_1] = {"ADC1", 1, 1, 0, MEC1322_ADC_CH(1)}, + [ADC_CH_2] = {"ADC2", 1, 1, 0, MEC1322_ADC_CH(2)}, + [ADC_CH_3] = {"ADC3", 1, 1, 0, MEC1322_ADC_CH(3)}, + [ADC_CH_4] = {"ADC4", 1, 1, 0, MEC1322_ADC_CH(4)}, +}; +BUILD_ASSERT(ARRAY_SIZE(adc_channels) == ADC_CH_COUNT); + /* Physical fans. These are logically separate from pwm_channels. */ const struct fan_t fans[] = { {.flags = FAN_USE_RPM_MODE, diff --git a/board/mec1322_evb/board.h b/board/mec1322_evb/board.h index fa4f53609c..a4142a9fdc 100644 --- a/board/mec1322_evb/board.h +++ b/board/mec1322_evb/board.h @@ -12,6 +12,7 @@ #define CONFIG_SYSTEM_UNLOCKED /* Allow dangerous commands */ #define CONFIG_WATCHDOG_HELP #define CONFIG_FANS 1 +#define CONFIG_ADC /* Modules we want to exclude */ #undef CONFIG_EEPROM @@ -23,6 +24,15 @@ #ifndef __ASSEMBLER__ +enum adc_channel { + ADC_CH_1 = 0, + ADC_CH_2, + ADC_CH_3, + ADC_CH_4, + + ADC_CH_COUNT +}; + /* GPIO signal list */ enum gpio_signal { GPIO_LED1 = 0, diff --git a/chip/mec1322/adc.c b/chip/mec1322/adc.c new file mode 100644 index 0000000000..e000e01cce --- /dev/null +++ b/chip/mec1322/adc.c @@ -0,0 +1,107 @@ +/* Copyright (c) 2013 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. + */ + +#include "adc.h" +#include "adc_chip.h" +#include "common.h" +#include "console.h" +#include "hooks.h" +#include "registers.h" +#include "task.h" +#include "timer.h" +#include "util.h" + +/* + * Conversion on a single channel takes less than 12 ms. Set timeout to + * 15 ms so that we have a 3-ms margin. + */ +#define ADC_SINGLE_READ_TIME 15000 + +struct mutex adc_lock; + +static task_id_t task_waiting; + +static int start_single_and_wait(int timeout) +{ + int event; + + task_waiting = task_get_current(); + + /* Start conversion */ + MEC1322_ADC_CTRL |= 1 << 1; + + /* Wait for interrupt */ + event = task_wait_event(timeout); + task_waiting = TASK_ID_INVALID; + return event != TASK_EVENT_TIMER; +} + +int adc_read_channel(enum adc_channel ch) +{ + const struct adc_t *adc = adc_channels + ch; + int value; + + mutex_lock(&adc_lock); + + MEC1322_ADC_SINGLE = 1 << adc->channel; + + if (start_single_and_wait(ADC_SINGLE_READ_TIME)) + value = MEC1322_ADC_READ(adc->channel) * adc->factor_mul / + adc->factor_div + adc->shift; + else + value = ADC_READ_ERROR; + + mutex_unlock(&adc_lock); + return value; +} + +int adc_read_all_channels(int *data) +{ + int i; + int ret = EC_SUCCESS; + const struct adc_t *adc; + + mutex_lock(&adc_lock); + + MEC1322_ADC_SINGLE = 0; + for (i = 0; i < ADC_CH_COUNT; ++i) + MEC1322_ADC_SINGLE |= 1 << adc_channels[i].channel; + + if (!start_single_and_wait(ADC_SINGLE_READ_TIME * ADC_CH_COUNT)) + goto exit_all_channels; + + for (i = 0; i < ADC_CH_COUNT; ++i) { + adc = adc_channels + i; + data[i] = MEC1322_ADC_READ(adc->channel) * adc->factor_mul / + adc->factor_div + adc->shift; + } + +exit_all_channels: + mutex_unlock(&adc_lock); + return ret; +} + +static void adc_init(void) +{ + /* Activate ADC module */ + MEC1322_ADC_CTRL |= 1 << 0; + + /* Enable interrupt */ + task_waiting = TASK_ID_INVALID; + MEC1322_INT_ENABLE(17) |= 1 << 10; + MEC1322_INT_BLK_EN |= 1 << 17; + task_enable_irq(MEC1322_IRQ_ADC_SNGL); +} +DECLARE_HOOK(HOOK_INIT, adc_init, HOOK_PRIO_DEFAULT); + +static void adc_interrupt(void) +{ + /* Clear interrupt status bit */ + MEC1322_ADC_CTRL |= 1 << 7; + + if (task_waiting != TASK_ID_INVALID) + task_wake(task_waiting); +} +DECLARE_IRQ(MEC1322_IRQ_ADC_SNGL, adc_interrupt, 2); diff --git a/chip/mec1322/adc_chip.h b/chip/mec1322/adc_chip.h new file mode 100644 index 0000000000..37e7f477e6 --- /dev/null +++ b/chip/mec1322/adc_chip.h @@ -0,0 +1,33 @@ +/* Copyright (c) 2013 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. + */ + +/* MEC1322-specific ADC module for Chrome EC */ + +#ifndef __CROS_EC_ADC_CHIP_H +#define __CROS_EC_ADC_CHIP_H + +/* Data structure to define ADC channels. */ +struct adc_t { + const char *name; + int factor_mul; + int factor_div; + int shift; + int channel; +}; + +/* + * Boards must provide this list of ADC channel definitions. This must match + * the enum adc_channel list provided by the board. + */ +extern const struct adc_t adc_channels[]; + +/* Minimum and maximum values returned by adc_read_channel(). */ +#define ADC_READ_MIN 0 +#define ADC_READ_MAX 1023 + +/* Just plain id mapping for code readability */ +#define MEC1322_ADC_CH(x) (x) + +#endif /* __CROS_EC_ADC_CHIP_H */ diff --git a/chip/mec1322/build.mk b/chip/mec1322/build.mk index 60376d1a5f..5481581ebe 100644 --- a/chip/mec1322/build.mk +++ b/chip/mec1322/build.mk @@ -13,6 +13,7 @@ CFLAGS_CPU+=-march=armv7e-m -mcpu=cortex-m4 # Required chip modules chip-y=clock.o gpio.o hwtimer.o system.o uart.o jtag.o +chip-$(CONFIG_ADC)+=adc.o chip-$(CONFIG_FANS)+=fan.o chip-$(CONFIG_I2C)+=i2c.o chip-$(CONFIG_LPC)+=lpc.o diff --git a/chip/mec1322/registers.h b/chip/mec1322/registers.h index 14c1aebfff..5a5aa98f2b 100644 --- a/chip/mec1322/registers.h +++ b/chip/mec1322/registers.h @@ -262,6 +262,16 @@ static inline uintptr_t gpio_port_base(int port_id) #define MEC1322_KS_EXT_CTRL REG32(MEC1322_KS_BASE + 0x14) +/* ADC */ +#define MEC1322_ADC_BASE 0x40007c00 +#define MEC1322_ADC_CTRL REG32(MEC1322_ADC_BASE + 0x0) +#define MEC1322_ADC_DELAY REG32(MEC1322_ADC_BASE + 0x4) +#define MEC1322_ADC_STS REG32(MEC1322_ADC_BASE + 0x8) +#define MEC1322_ADC_SINGLE REG32(MEC1322_ADC_BASE + 0xc) +#define MEC1322_ADC_REPEAT REG32(MEC1322_ADC_BASE + 0x10) +#define MEC1322_ADC_READ(x) REG32(MEC1322_ADC_BASE + 0x14 + (x) * 0x4) + + /* IRQ Numbers */ #define MEC1322_IRQ_I2C_0 0 #define MEC1322_IRQ_I2C_1 1 |