knx/examples/knx-cc1310/coresdk_cc13xx_cc26xx/source/ti/drivers/adc/ADCCC26XX.c
nanosonde d90843ba45
Add support for CC1310 platform based on SimpleLink SDK (#94)
* Initial commit

* Clean up

* Remove display code

* Change cmake build

* Add SimpleLink SDK for CC13xx/CC26xx as submodule

* Remove commented line from build.sh

* Working build

* Remove SDK submodule

* Squashed 'examples/knx-cc1310/coresdk_cc13xx_cc26xx/' content from commit 0d78d32

git-subtree-dir: examples/knx-cc1310/coresdk_cc13xx_cc26xx
git-subtree-split: 0d78d3280357416a5c0388148cda13717c9ffaa5

* Add more comments and enable Power_idleFunc() for NoRTOS variant.
Internal SDK driver functions which have to wait for something
will cause Power_idleFunc to be called instead of doing busy wait.

* Move CC1310 platform init around

* Optimize a bit more in debug build config as the binary does not fit into 128Kb flash otherwise.

* Explicitly list each source/header file in build config. Use linker group to resolve circular dependencies.

* Ignore vscode settings.json

* Increase stacks size

* Only compile CC1310 source code if #define DeviceFamily_CC13X0

* initial commit of CC1310 RF driver with first working RX version

* Better handling of buttonUp() across platforms

* Start cleanup

* continue cleanup

* Fix bau2920 compilation

* Continue cleanup

* Fix compilation in other examples

* Fix compilation

* htons() and ntohs() only for SAMD and STM32, but not for Linux and ESP8266 and ESP32

* htons(9 and ntohs() needed for CC13x0

* Continue cleanup

* Add CC1310 platform to CI

* Fix CI

* Use more recent toolchain from ARM

* Fix travis

* Use Ubuntu Focal

* Fix toolchain for travis

* Fix package name

* Fix toolchain

* Add libstdc++-dev package

* Add newlib packages

* Remove commented commands from CI

* Fix travis

* Fix compilation of knxPython

* Clean up linefeeds

* Fix RX callback

* Move RF CRC16-DNP to bits.cpp

* Fix TX

* Optimization: do not calc CRC for block1 again in rf_data_link_layer

* Make newline optional in printHex

* Cleanup. First working version: ETS5 programming of individual address via KNX/RF coupler.

* Use LEDs and Buttons to control ProgMode and Flash Erase

* Remove settings.json (VScode)

* Add README.md

* Update README.md

* Update README.md

* Fix typo
2020-11-10 21:52:38 +01:00

372 lines
12 KiB
C

/*
* Copyright (c) 2016-2018, Texas Instruments Incorporated
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* * Neither the name of Texas Instruments Incorporated nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include <stdint.h>
/* Kernel services */
#include <ti/drivers/dpl/DebugP.h>
#include <ti/drivers/dpl/SemaphoreP.h>
#include <ti/drivers/dpl/HwiP.h>
/* TI-RTOS drivers */
#include <ti/drivers/ADC.h>
#include <ti/drivers/adc/ADCCC26XX.h>
#include <ti/drivers/PIN.h>
#include <ti/drivers/pin/PINCC26XX.h>
#include <ti/drivers/Power.h>
#include <ti/drivers/power/PowerCC26XX.h>
#include <ti/devices/DeviceFamily.h>
#include DeviceFamily_constructPath(inc/hw_memmap.h)
#include DeviceFamily_constructPath(inc/hw_ints.h)
#include DeviceFamily_constructPath(inc/hw_types.h)
#include DeviceFamily_constructPath(inc/hw_aux_evctl.h)
#include DeviceFamily_constructPath(driverlib/aux_adc.h)
#include DeviceFamily_constructPath(driverlib/aux_smph.h)
#include DeviceFamily_constructPath(driverlib/sys_ctrl.h)
#include DeviceFamily_constructPath(driverlib/ioc.h)
#include DeviceFamily_constructPath(driverlib/aon_ioc.h)
#if (DeviceFamily_PARENT == DeviceFamily_PARENT_CC13X0_CC26X0)
#include DeviceFamily_constructPath(driverlib/aux_wuc.h)
#elif (DeviceFamily_PARENT == DeviceFamily_PARENT_CC13X2_CC26X2)
#define AUX_EVCTL_EVTOMCUFLAGS_ADC_DONE AUX_EVCTL_EVTOMCUFLAGS_AUX_ADC_DONE
#define AUX_EVCTL_EVTOMCUFLAGS_ADC_IRQ AUX_EVCTL_EVTOMCUFLAGS_AUX_ADC_IRQ
#endif
/*
* =============================================================================
* Public Function Declarations
* =============================================================================
*/
void ADCCC26XX_close(ADC_Handle handle);
void ADCCC26XX_init(ADC_Handle handle);
ADC_Handle ADCCC26XX_open(ADC_Handle handle, ADC_Params *params);
int_fast16_t ADCCC26XX_convert(ADC_Handle handle, uint16_t *value);
int_fast16_t ADCCC26XX_control(ADC_Handle handle, uint_fast16_t cmd, void *arg);
uint32_t ADCCC26XX_convertToMicroVolts(ADC_Handle handle, uint16_t adcValue);
/*
* =============================================================================
* Private Function Declarations
* =============================================================================
*/
/*
* =============================================================================
* Constants
* =============================================================================
*/
/* ADC function table for ADCCC26XX implementation */
const ADC_FxnTable ADCCC26XX_fxnTable = {
ADCCC26XX_close,
ADCCC26XX_control,
ADCCC26XX_convert,
ADCCC26XX_convertToMicroVolts,
ADCCC26XX_init,
ADCCC26XX_open
};
/*
* =============================================================================
* Private Global Variables
* =============================================================================
*/
/* Keep track the adc handle instance to create and delete adcSemaphore */
static uint16_t adcInstance = 0;
/* Semaphore to arbitrate access to the single ADC peripheral between multiple handles */
static SemaphoreP_Struct adcSemaphore;
/*
* =============================================================================
* Function Definitions
* =============================================================================
*/
/*
* ======== ADCCC26XX_close ========
*/
void ADCCC26XX_close(ADC_Handle handle){
ADCCC26XX_Object *object;
DebugP_assert(handle);
object = handle->object;
uint32_t key = HwiP_disable();
if (object->isOpen) {
adcInstance--;
if (adcInstance == 0) {
SemaphoreP_destruct(&adcSemaphore);
}
DebugP_log0("ADC: Object closed");
}
else {
return;
}
object->isOpen = false;
HwiP_restore(key);
/* Deallocate pins */
if (object->pinHandle){
PIN_close(object->pinHandle);
}
}
/*
* ======== ADCCC26XX_control ========
*/
int_fast16_t ADCCC26XX_control(ADC_Handle handle, uint_fast16_t cmd, void *arg){
/* No implementation yet */
return ADC_STATUS_UNDEFINEDCMD;
}
/*
* ======== ADCCC26XX_convert ========
*/
int_fast16_t ADCCC26XX_convert(ADC_Handle handle, uint16_t *value){
ADCCC26XX_HWAttrs const *hwAttrs;
ADCCC26XX_Object *object;
int_fast16_t conversionResult = ADC_STATUS_ERROR;
uint16_t conversionValue = 0;
uint32_t interruptStatus = 0;
DebugP_assert(handle);
/* Get handle */
hwAttrs = handle->hwAttrs;
/* Get the object */
object = handle->object;
if (object->isProtected) {
/* Acquire the lock for this particular ADC handle */
SemaphoreP_pend(&adcSemaphore, SemaphoreP_WAIT_FOREVER);
}
/* Set constraints to guarantee operation */
Power_setConstraint(PowerCC26XX_DISALLOW_STANDBY);
/* Acquire the ADC hw semaphore. Return an error if the hw semaphore is not
* available. There is only one interrupt available for the hw semaphores and it
* is used by the TDC already. Busy-wait polling might lock up the device and starting
* timeout clocks would add overhead and be clunky. It is better if such functionality
* is implemented at application level if desired.
*/
if(!AUXSMPHTryAcquire(AUX_SMPH_2)){
Power_releaseConstraint(PowerCC26XX_DISALLOW_STANDBY);
if (object->isProtected) {
SemaphoreP_post(&adcSemaphore);
}
return conversionResult;
}
/* Specify input in ADC module */
AUXADCSelectInput(hwAttrs->adcCompBInput);
/* Flush the ADC FIFO in case we have triggered prior to this call */
AUXADCFlushFifo();
/* If input scaling is set to disabled in the params, disable it */
if (!hwAttrs->inputScalingEnabled){
AUXADCDisableInputScaling();
}
/* Use synchronous sampling mode and prepare for trigger */
AUXADCEnableSync(hwAttrs->refSource, hwAttrs->samplingDuration, hwAttrs->triggerSource);
/* Manually trigger the ADC once */
AUXADCGenManualTrigger();
/* Poll until the sample is ready */
conversionValue = AUXADCReadFifo();
/* Get the status of the ADC_IRQ line and ADC_DONE.
* Despite not using the interrupt line, we need to clear it so that the
* ADCBuf driver does not call Hwi_construct and have thte interrupt fire
* immediately.
*/
interruptStatus = HWREG(AUX_EVCTL_BASE + AUX_EVCTL_O_EVTOMCUFLAGS) &
(AUX_EVCTL_EVTOMCUFLAGS_ADC_IRQ |
AUX_EVCTL_EVTOMCUFLAGS_ADC_DONE);
/* Clear the ADC_IRQ flag in AUX_EVTCTL */
HWREG(AUX_EVCTL_BASE + AUX_EVCTL_O_EVTOMCUFLAGSCLR) = interruptStatus;
/* Clear the ADC_IRQ within the NVIC as AUX_EVTCTL will only set the
* relevant flag in the NVIC it will not clear it.
*/
HwiP_clearInterrupt(INT_AUX_ADC_IRQ);
conversionResult = ADC_STATUS_SUCCESS;
/* Shut down the ADC peripheral */
AUXADCDisable();
/* Release the ADC hw semaphore */
AUXSMPHRelease(AUX_SMPH_2);
/* Allow entering standby again after ADC conversion complete */
Power_releaseConstraint(PowerCC26XX_DISALLOW_STANDBY);
if (object->isProtected) {
/* Release the lock for this particular ADC handle */
SemaphoreP_post(&adcSemaphore);
}
/* If we want to return the trimmed value, calculate it here. */
if (hwAttrs->returnAdjustedVal) {
uint32_t gain = AUXADCGetAdjustmentGain(hwAttrs->refSource);
uint32_t offset = AUXADCGetAdjustmentOffset(hwAttrs->refSource);
conversionValue = AUXADCAdjustValueForGainAndOffset(conversionValue, gain, offset);
}
*value = conversionValue;
/* Return the number of bytes transfered by the ADC */
return conversionResult;
}
/*
* ======== ADCCC26XX_convertToMicroVolts ========
*/
uint32_t ADCCC26XX_convertToMicroVolts(ADC_Handle handle, uint16_t adcValue){
ADCCC26XX_HWAttrs const *hwAttrs;
uint32_t adjustedValue;
DebugP_assert(handle);
/* Get the pointer to the hwAttrs */
hwAttrs = handle->hwAttrs;
/* Only apply trim if specified*/
if (hwAttrs->returnAdjustedVal) {
adjustedValue = adcValue;
}
else {
uint32_t gain = AUXADCGetAdjustmentGain(hwAttrs->refSource);
uint32_t offset = AUXADCGetAdjustmentOffset(hwAttrs->refSource);
adjustedValue = AUXADCAdjustValueForGainAndOffset(adcValue, gain, offset);
}
return AUXADCValueToMicrovolts((hwAttrs->inputScalingEnabled ? AUXADC_FIXED_REF_VOLTAGE_NORMAL : AUXADC_FIXED_REF_VOLTAGE_UNSCALED), adjustedValue);
}
/*
* ======== ADCCC26XX_init ========
*/
void ADCCC26XX_init(ADC_Handle handle){
ADCCC26XX_Object *object;
/* Get the object */
object = handle->object;
/* Mark the object as available */
object->isOpen = false;
}
/*
* ======== ADCCC26XX_open ========
*/
ADC_Handle ADCCC26XX_open(ADC_Handle handle, ADC_Params *params){
ADCCC26XX_Object *object;
ADCCC26XX_HWAttrs const *hwAttrs;
PIN_Config adcPinTable[2];
DebugP_assert(handle);
/* Get object and hwAttrs */
object = handle->object;
hwAttrs = handle->hwAttrs;
/* Determine if the driver was already opened */
uint32_t key = HwiP_disable();
if (object->isOpen){
DebugP_log0("ADC: Error! Already in use.");
HwiP_restore(key);
return NULL;
}
object->isOpen = true;
/* remember thread safety protection setting */
object->isProtected = params->isProtected;
/* If this is the first handle requested, set up the semaphore as well */
if (adcInstance == 0) {
/* Setup semaphore */
SemaphoreP_constructBinary(&adcSemaphore, 1);
}
adcInstance++;
/* On Chameleon, ANAIF must be clocked to use it. On Agama, the register inferface is always available. */
#if (DeviceFamily_PARENT == DeviceFamily_PARENT_CC13X0_CC26X0)
/* Turn on the ANAIF clock. ANIAF contains the aux ADC. */
AUXWUCClockEnable(AUX_WUC_ANAIF_CLOCK);
AUXWUCClockEnable(AUX_WUC_ADI_CLOCK);
#endif
HwiP_restore(key);
/* Reserve the DIO defined in the hwAttrs */
uint8_t i = 0;
/* Add pin to measure on */
adcPinTable[i++] = hwAttrs->adcDIO |
PIN_NOPULL |
PIN_INPUT_DIS |
PIN_GPIO_OUTPUT_DIS |
PIN_IRQ_DIS |
PIN_DRVSTR_MIN;
/* Terminate pin list */
adcPinTable[i] = PIN_TERMINATE;
object->pinHandle = PIN_open(&object->pinState, adcPinTable);
if (!object->pinHandle){
DebugP_log0("ADC: Error! Already in use.");
object->isOpen = false;
return NULL;
}
DebugP_log0("ADC: Object opened");
return handle;
}