/* * This file is part of the Micro Python project, http://micropython.org/ * * The MIT License (MIT) * * Copyright (c) 2017 Dan Halbert for Adafruit Industries * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #include "hal/include/hal_adc_sync.h" #include "hpl/gclk/hpl_gclk_base.h" #include "hri/hri_mclk_d51.h" // The clock initializer values are rather random, so we need to put them in // tables for lookup. We can't compute them. static const uint8_t SERCOMx_GCLK_ID_CORE[] = { SERCOM0_GCLK_ID_CORE, SERCOM1_GCLK_ID_CORE, SERCOM2_GCLK_ID_CORE, SERCOM3_GCLK_ID_CORE, SERCOM4_GCLK_ID_CORE, SERCOM5_GCLK_ID_CORE, #ifdef SERCOM6 SERCOM6_GCLK_ID_CORE, #endif #ifdef SERCOM7 SERCOM7_GCLK_ID_CORE, #endif }; static const uint8_t SERCOMx_GCLK_ID_SLOW[] = { SERCOM0_GCLK_ID_SLOW, SERCOM1_GCLK_ID_SLOW, SERCOM2_GCLK_ID_SLOW, SERCOM3_GCLK_ID_SLOW, SERCOM4_GCLK_ID_SLOW, SERCOM5_GCLK_ID_SLOW, #ifdef SERCOM6 SERCOM6_GCLK_ID_SLOW, #endif #ifdef SERCOM7 SERCOM7_GCLK_ID_SLOW, #endif }; Sercom* sercom_insts[SERCOM_INST_NUM] = SERCOM_INSTS; // Clock initialization as done in Atmel START. void samd_peripherals_sercom_clock_init(Sercom* sercom, uint8_t sercom_index) { hri_gclk_write_PCHCTRL_reg(GCLK, SERCOMx_GCLK_ID_CORE[sercom_index], GCLK_PCHCTRL_GEN_GCLK1_Val | (1 << GCLK_PCHCTRL_CHEN_Pos)); hri_gclk_write_PCHCTRL_reg(GCLK, SERCOMx_GCLK_ID_SLOW[sercom_index], GCLK_PCHCTRL_GEN_GCLK3_Val | (1 << GCLK_PCHCTRL_CHEN_Pos)); // hri_mclk_set_APBAMASK_SERCOMx_bit is an inline, so let's use a switch, not a table. switch (sercom_index) { case 0: hri_mclk_set_APBAMASK_SERCOM0_bit(MCLK); break; case 1: hri_mclk_set_APBAMASK_SERCOM1_bit(MCLK); break; case 2: hri_mclk_set_APBBMASK_SERCOM2_bit(MCLK); break; case 3: hri_mclk_set_APBBMASK_SERCOM3_bit(MCLK); break; case 4: hri_mclk_set_APBDMASK_SERCOM4_bit(MCLK); break; case 5: hri_mclk_set_APBDMASK_SERCOM5_bit(MCLK); break; #ifdef SERCOM6 case 6: hri_mclk_set_APBDMASK_SERCOM6_bit(MCLK); break; #endif #ifdef SERCOM7 case 7: hri_mclk_set_APBDMASK_SERCOM7_bit(MCLK); break; #endif } } // Figure out the DOPO value given the chosen clock pad and mosi pad. // Return an out-of-range value (255) if the combination is not permitted // The ASF4 config files list this, but the SAMD51 datasheet // says 0x1 and 0x3 are reserved, so don't allow pad 3 SCK. // Transmit Data Pinout // <0x0=>PAD[0,1]_DO_SCK // <0x1=>PAD[2,3]_DO_SCK [RESERVED] // <0x2=>PAD[3,1]_DO_SCK // <0x3=>PAD[0,3]_DO_SCK [RESERVED] uint8_t samd_peripherals_get_spi_dopo(uint8_t clock_pad, uint8_t mosi_pad) { if (clock_pad != 1) { return 255; } if (mosi_pad == 0) { return 0x1; } if (mosi_pad == 3) { return 0x2; } return 255; } bool samd_peripherals_valid_spi_clock_pad(uint8_t clock_pad) { return clock_pad == 1; } // Do initialization and calibration setup needed for any use of the ADC. // The reference and resolution should be set by the caller. void samd_peripherals_adc_setup(struct adc_sync_descriptor *adc, Adc *instance) { // Turn the clocks on. if (instance == ADC0) { hri_mclk_set_APBDMASK_ADC0_bit(MCLK); hri_gclk_write_PCHCTRL_reg(GCLK, ADC0_GCLK_ID, GCLK_PCHCTRL_GEN_GCLK1_Val | (1 << GCLK_PCHCTRL_CHEN_Pos)); } else if (instance == ADC1) { hri_mclk_set_APBDMASK_ADC1_bit(MCLK); hri_gclk_write_PCHCTRL_reg(GCLK, ADC1_GCLK_ID, GCLK_PCHCTRL_GEN_GCLK1_Val | (1 << GCLK_PCHCTRL_CHEN_Pos)); } adc_sync_init(adc, instance, (void *)NULL); // SAMD51 has a CALIB register but doesn't have documented fuses for them. uint8_t biasrefbuf; uint8_t biasr2r; uint8_t biascomp; if (instance == ADC0) { biasrefbuf = ((*(uint32_t*) ADC0_FUSES_BIASREFBUF_ADDR) & ADC0_FUSES_BIASREFBUF_Msk) >> ADC0_FUSES_BIASREFBUF_Pos; biasr2r = ((*(uint32_t*) ADC0_FUSES_BIASR2R_ADDR) & ADC0_FUSES_BIASR2R_Msk) >> ADC0_FUSES_BIASR2R_Pos; biascomp = ((*(uint32_t*) ADC0_FUSES_BIASCOMP_ADDR) & ADC0_FUSES_BIASCOMP_Msk) >> ADC0_FUSES_BIASCOMP_Pos; } else { biasrefbuf = ((*(uint32_t*) ADC1_FUSES_BIASREFBUF_ADDR) & ADC1_FUSES_BIASREFBUF_Msk) >> ADC1_FUSES_BIASREFBUF_Pos; biasr2r = ((*(uint32_t*) ADC1_FUSES_BIASR2R_ADDR) & ADC1_FUSES_BIASR2R_Msk) >> ADC1_FUSES_BIASR2R_Pos; biascomp = ((*(uint32_t*) ADC1_FUSES_BIASCOMP_ADDR) & ADC1_FUSES_BIASCOMP_Msk) >> ADC1_FUSES_BIASCOMP_Pos; } hri_adc_write_CALIB_BIASREFBUF_bf(instance, biasrefbuf); hri_adc_write_CALIB_BIASR2R_bf(instance, biasr2r); hri_adc_write_CALIB_BIASCOMP_bf(instance, biascomp); }