From 2891f5e3a384851d5918b2955ab45fe2344574a2 Mon Sep 17 00:00:00 2001 From: "Spence Konde (aka Dr. Azzy)" Date: Tue, 17 Dec 2024 13:30:36 -0500 Subject: [PATCH 1/3] Fix #1165' --- megaavr/cores/megatinycore/core_devices.h | 54 ++++++++++----------- megaavr/cores/megatinycore/wiring_digital.c | 18 ++++--- 2 files changed, 37 insertions(+), 35 deletions(-) diff --git a/megaavr/cores/megatinycore/core_devices.h b/megaavr/cores/megatinycore/core_devices.h index bacf1326..f1a5cfd9 100644 --- a/megaavr/cores/megatinycore/core_devices.h +++ b/megaavr/cores/megatinycore/core_devices.h @@ -282,6 +282,7 @@ #define __AVR_TINY_2__ #endif +/* FOLLOWING THIS, SHARED WITH DxCoRE DIRECTLY */ #if defined(AC2) #define _AVR_AC_COUNT (3) #elif defined(AC1) @@ -585,6 +586,10 @@ #endif #endif + +/* End block shared with DxCore */ + + // Notice NUM_TOTAL_PORTS should always be 1, 2, 3, 6 or 7 - it is the number of the highest port plus 1. // The tinyAVRs have either just PORTA, PORTA and PORTB, or ports A, B, and C - 1, 2, or 3 ports. // The Dx and Ex parts,however, even on the 14-pin version, have representatives from ports A, C, D, and F. @@ -600,32 +605,27 @@ */ // bits are 0bRRRPPPPE - Reserved x3, prescale x4, prescale enable - nothing we need to preserve! -#define _setPrescale1x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (0))) -#define _setPrescale2x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, ( CLKCTRL_PDIV_2X_gc | CLKCTRL_PEN_bm))) /* 0x01 */ -#define _setPrescale4x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, ( CLKCTRL_PDIV_4X_gc | CLKCTRL_PEN_bm))) /* 0x03 */ -#define _setPrescale8x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, ( CLKCTRL_PDIV_8X_gc | CLKCTRL_PEN_bm))) /* 0x05 */ -#define _setPrescale16x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_16X_gc | CLKCTRL_PEN_bm))) /* 0x07 */ -#define _setPrescale32x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_32X_gc | CLKCTRL_PEN_bm))) /* 0x09 */ -#define _setPrescale64x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_64X_gc | CLKCTRL_PEN_bm))) /* 0x0B */ -#define _setPrescale6x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, ( CLKCTRL_PDIV_6X_gc | CLKCTRL_PEN_bm))) /* 0x11 */ -#define _setPrescale10x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_10X_gc | CLKCTRL_PEN_bm))) /* 0x13 */ -#define _setPrescale12x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_12X_gc | CLKCTRL_PEN_bm))) /* 0x15 */ -#define _setPrescale24x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_24X_gc | CLKCTRL_PEN_bm))) /* 0x17 */ -#define _setPrescale48x() (_PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_48X_gc | CLKCTRL_PEN_bm))) /* 0x19 */ - -/* -uint8_t _clockprescalers[] = {1, 2, 4, 8, 16, 32, 64, 6, 10, 12, 24, 48}; -uint8_t _clockprescalesettings[] = {0, 1, 3, 5, 7, 9, B, 17, 19, 21, 23, 25}; - -int8_t _setPrescale(int8_t prescale) { - for (x = 0x00; x < 11; x++) { - if (_clockprescalers[x] == prescale) { - _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, _clockprescalesettings[prescale]); - return prescale; - } - } - return -1; // invalid prescaler passed -} +#define _setPrescale1x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (0)) +#define _setPrescale2x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, ( CLKCTRL_PDIV_2X_gc | CLKCTRL_PEN_bm)) /* 0x01 */ +#define _setPrescale4x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, ( CLKCTRL_PDIV_4X_gc | CLKCTRL_PEN_bm)) /* 0x03 */ +#define _setPrescale8x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, ( CLKCTRL_PDIV_8X_gc | CLKCTRL_PEN_bm)) /* 0x05 */ +#define _setPrescale16x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_16X_gc | CLKCTRL_PEN_bm)) /* 0x07 */ +#define _setPrescale32x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_32X_gc | CLKCTRL_PEN_bm)) /* 0x09 */ +#define _setPrescale64x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_64X_gc | CLKCTRL_PEN_bm)) /* 0x0B */ +#define _setPrescale6x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, ( CLKCTRL_PDIV_6X_gc | CLKCTRL_PEN_bm)) /* 0x11 */ +#define _setPrescale10x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_10X_gc | CLKCTRL_PEN_bm)) /* 0x13 */ +#define _setPrescale12x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_12X_gc | CLKCTRL_PEN_bm)) /* 0x15 */ +#define _setPrescale24x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_24X_gc | CLKCTRL_PEN_bm)) /* 0x17 */ +#define _setPrescale48x() _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, (CLKCTRL_PDIV_48X_gc | CLKCTRL_PEN_bm)) /* 0x19 */ +/* DANGER DANGER - NOT MEANT FOR ENDUSER APPLICATIONS. GI - GO */ +#define _getPrescale() (CLKCTRL_MCLKCTRLB) +#define _setPrescaleValue(val) _PROTECTED_WRITE(CLKCTRL_MCLKCTRLB, val) +/* This is Yucky. These are macros. +Intended usage scenario: +uint8_t oldpsc=_getPrescale(); +setPrescaler10x(); +//dostuff +setPrescalerValue(oldpsc); */ #if MEGATINYCORE_SERIES <= 2 @@ -718,8 +718,6 @@ int8_t _setPrescale(int8_t prescale) { #endif -#define CLOCK_TUNE_START (USER_SIGNATURES_SIZE - 12) - /* Microchip has shown a tendency to rename registers bitfields and similar between product lines, even when the behavior is identical. * This is a major hindrance to writing highly portable code which I assume is what most people wish to do. It certainly beats having * to run code through find replace making trivial changes, forcing a fork where you would rather not have one. diff --git a/megaavr/cores/megatinycore/wiring_digital.c b/megaavr/cores/megatinycore/wiring_digital.c index b33db741..36ada722 100644 --- a/megaavr/cores/megatinycore/wiring_digital.c +++ b/megaavr/cores/megatinycore/wiring_digital.c @@ -121,14 +121,14 @@ void turnOffPWM(uint8_t pin) #if !defined(TCA_BUFFERED_3PIN) // uint8_t *timer_cmp_out; /* Bit position will give output channel */ - #ifdef __AVR_ATtinyxy2__ + #ifdef __AVR_ATtinyxy2__ //8 pin parts if (bit_mask == 0x80) { bit_mask = 1; // on the xy2, WO0 is on PA7 } if (bit_mask > 0x04) { // -> bit_pos > 2 -> output channel controlled by HCMP bit_mask <<= 1; // mind the gap (between LCMP and HCMP) } - #else + #else // Normal parts if (digitalPinToPort(pin) == PB) { // WO0-WO2, Bitmask has one of these bits 1: 0b00hhhlll. if (bit_mask > 0x04) { // Is it one of the three high ones? If so bit_mask <<= 1; // nudge it 1 place left swap nybbles since that's 1 clock faster than 3 rightshifts. @@ -138,13 +138,17 @@ void turnOffPWM(uint8_t pin) // Otherwise, it's WO3-5. These will always be on 0b00hhh000,. Here since we ARE working with a high half timer, we need to just leftshift it once. bit_mask <<= 1; } - #endif + #endif // End if 8 vs normal parts TCA0.SPLIT.CTRLB &= ~bit_mask; - #else // 3-pin mode. Means we know it's on PORTB} - if (bit_mask > 0x04) { // Is it one of the three high ones? If so - bit_mask <<= 1; // nudge it 1 place left swap nybbles since that's 1 clock faster than 3 rightshifts. - _SWAP(bit_mask); // swap nybbles since that's 1 clock faster than 3 rightshifts. + #else // 3-pin mode. Means we know it's either in set A or B: 0b0AAAxBBB We get 0x00ABC + if(bit_mask & 0b00111000) { + bit_mask <<= 1; // 0b01110000 + } else { + //bit_maslk <<= 3; same result, but in 3 clocks 3 words instead of 1 and 1! + + _SWAP(bit_mask); // 0b0111000 } + TCA0.SINGLE.CTRLB &= ~bit_mask; #endif break; } From 17905299b47ded73af36d13317e074fae35da96a Mon Sep 17 00:00:00 2001 From: "Spence Konde (aka Dr. Azzy)" Date: Tue, 17 Dec 2024 14:00:04 -0500 Subject: [PATCH 2/3] Update dirty_tricks.h --- megaavr/cores/megatinycore/dirty_tricks.h | 44 +++++++++++++++-------- 1 file changed, 30 insertions(+), 14 deletions(-) diff --git a/megaavr/cores/megatinycore/dirty_tricks.h b/megaavr/cores/megatinycore/dirty_tricks.h index 0125bdde..d349798a 100644 --- a/megaavr/cores/megatinycore/dirty_tricks.h +++ b/megaavr/cores/megatinycore/dirty_tricks.h @@ -5,11 +5,15 @@ * See LICENSE.txt for full legal boilerplate if you must */ -/* Okay... This was getting fucking absurd. +/* Okay... This was getting absurd. We had too many piecemeal hacks to work around shit compiler output, the poor quality of which is was apparent immediately upon + * examination of the assembler listings early on, in code that was everywhere, and hence which I had to stare at any time I debugged anything. * We (well, me and the one, maybe two other people here who speak fluent avr asm) all knew we needed unholy tricks like this. - * We wish we didn't. But while the GCC part of avr-gcc does a pretty thorough and consistent job optimizing... + * We wish we didn't. But while the GCC part of avr-gcc does a pretty thorough and consistent job optimizing.. * The AVR part is piss-poor. It's fucking awful. As I understand it, GCC from a big picture, first transforms the code into an idealized - * version that has standins for registers, but "almost" maps to machine instructions, but is universal and this is where the bulk of the optimization happens + * version that has standins for registers and is architecture agnostic, but tends to correspond very very nearly with machine instructions. Because there is + * a massive amount of stuff compiled with GCC and other tools built on the same foundations (they're the only game in town if aren't willing to use a proprietary + * one. I don't really understand the appeal of those... ) considerable effort has been directed towards improvement, in + * order to improve efficiency of high profile computing applications * This happens in many passes to perform optimization that isn't machine specific (and hence have gotten resources lavished on them by everyone). * Then, it has to go from that ideal world to reality, and start naming instructions and assigning registers. This is machine specific. * And it is here that AVR hasn't gotten much love. The result is sometimes what might be termed pathological assembly (as in, it is indicative disease). @@ -44,15 +48,16 @@ * * With this update, we are centralizing and documenting all the dirty assembler tricks. * - * I know that lowercase names and macros are sinful. Yes these are macros. Except for the classics (and we have appropriate duplicated of those), they are now lowercase. + * I know that mixing lowercase names and macros is sinful. Yes these are macros. Except for the classics (and we have appropriate duplicated of those), they are now lowercase. * I know macros are supposed to be all uppercase. But we're prefixing them with an underscore, so you still know there's spooky shit involved, it's just * too unpleasant to type and too ugly to read to have all these longass macro names, making a mess of all our core "speed matters" code - and besides that, - * while macros generally don't care what types they get passed, these totally will cast the arguments to specific types like functions do. Except more permissively - - * that's an explicit cast, which will eat things that functions won't implicitly cast), but without actually changing the type of anything :-P + * while macros generally don't care what types they get passed, these totally will cast the arguments to specific types like functions do (actually, significantly more vigorously) * * General Advice: * Don't use any of the wacky ones unless you know what the fuck you're doing. This is largely for internal use within the core, because we do, and we know that - * lots of people who dont and couldn't reproduce this performance enhancement themself if they needed it use the core. So where there's a high impact + * lots of people don't know nor want to know asm (even normal, polite asm, which this often isn't), and thus couldn't reproduce this performance enhancement + * if they needed or wanted it, and that describes most people who use the core. Essentially, I feel that when I see a way to silently make the core perform better, even if it looks + * heinous, do it, because if you don't, nobody else will, and surely someomne out there needs a bit more performance right? * bit of code - either repeated a lot, and/or in a time-critical section of code that benefits from doing unholy things like this, we're going to do it to make * the core better. * @@ -80,6 +85,7 @@ * * And now all the hideously dirty ones: * In all cases, pointer-like 16-bit values should be a local variable, either a pointer or uint16_t. + * These "tricks" allow us to * _addLow(uint16_t a, uint8_t b) * _subLow(uint16_t a, uint8_t b) * _addHigh(uint16_t a, uint8_t b) @@ -137,12 +143,7 @@ Not enabled. Ugly ways to get delays at very small flash cost. // Really, these are things you shoudnt do unless you have your back against the flash/RAM limits and a gun to your head. #endif */ -#ifndef _NOP14 - #define _NOP14() __asm__ __volatile__ ("rjmp .+2" "\n\t" /* same idea as above. */ \ - "ret" "\n\t" /* Except now it's no longer able to beat the loop if it has a free register. */ \ - "rcall .-4" "\n\t" /* so this is unlikely to be better, ever. */ \ - "rcall .-6" "\n\t" ) -#endif + /* Beyond this, just use a loop. * If you don't need submicrosecond accuracy, just use delayMicroseconds(), which uses very similar methods. See Ref_Timers @@ -190,6 +191,20 @@ Not enabled. Ugly ways to get delays at very small flash cost. /*********************/ /* Low and high Math */ /*********************/ +/* These are dirty, ugly hacky things. What is the point of them? They save a single clock each when used at a point where they are relevant and correct. + By prohibiting the crossing of a 256b barier (the high versions are less useful I daresay...), you save a clock cycle. Note that wraparound behavior is predictable + + A big part of this was just how often the stock code was doing this, over and over and over: + adiw r26, 10 + ld X, rwhatever + sbiw r26, 10 + adiw r26, 12 + ld X, rsomeother + sbit r26, 12 + ADIW and SBIW are both 2 clock instructions, and a load is 2, so that's 6 clocks; you can do it in 3n+1 instead of 6n clocks as long as they're in the same block of 256 addresses, which is often guaranteed. + Other times you may be subtracting values that are mathematically certain to never result in an over or underflow +*/ + #define _addLow(a,b) __asm__ __volatile__ ("add %0A, %1" "\n\t" :"+r"((uint16_t)(a)):"r"((uint8_t) b)) #define _subLow(a,b) __asm__ __volatile__ ("sub %0A, %1" "\n\t" :"+r"((uint16_t)(a)):"r"((uint8_t) b)) #define _addHigh(a,b) __asm__ __volatile__ ("add %0B, %1" "\n\t" :"+r"((uint16_t)(a)):"r"((uint8_t) b)) @@ -320,7 +335,8 @@ Not enabled. Ugly ways to get delays at very small flash cost. * SK: I've totally been desperate enough on a 16k part that I was replacing double-quoted strings 1 character long with single quoted characters * in order to get rid of the space the null took up and pulling strings from progmem to RAM because that saved a word vs printing it with F() macro * (it was a classic AVR, no memory mapped flash). I was told 500 of them were getting programmed the next day and we needed a feature added. One that I suspected - * weeks ago would be needed, but they'd finally realized it the evening before D-day at around 9pm with a few hundred byes of flash left. Good times! + * weeks ago would be needed, but they'd finally realized it the evening before D-day at around 9pm with a few hundred byes of flash left. Good times! Would have been cake if I knew ASM then + * since then I could optimize the code. * * _makeFastPtr() should be used when you are only reading from that one address, or consecutive addresses starting there (eg, *ptr++ or *--ptr), * so it can use the X, Y or Z register. From f47cb37ef5627e12230a96538b4f32fd5b37518e Mon Sep 17 00:00:00 2001 From: "Spence Konde (aka Dr. Azzy)" Date: Tue, 17 Dec 2024 15:22:09 -0500 Subject: [PATCH 3/3] Fixec #1158 correctly. --- megaavr/libraries/Wire/src/twi_pins.c | 381 +++++++++++++++++++++++++- 1 file changed, 380 insertions(+), 1 deletion(-) diff --git a/megaavr/libraries/Wire/src/twi_pins.c b/megaavr/libraries/Wire/src/twi_pins.c index 408eb73a..c01879d9 100644 --- a/megaavr/libraries/Wire/src/twi_pins.c +++ b/megaavr/libraries/Wire/src/twi_pins.c @@ -365,4 +365,383 @@ bool TWI0_swap(uint8_t state) { } #endif return false; -#endif \ No newline at end of file +} + + +void TWI0_usePullups() { + // make sure we don't get errata'ed: Clear the pins + // and then turn on the pullups. + #if defined(PORTMUX_TWIROUTEA) // Dx or Ex-series + uint8_t portmux = PORTMUX.TWIROUTEA & PORTMUX_TWI0_gm; + PORT_t *port; + // PORTA and PORTC are present on all parts with a TWIROUTEA register + if(portmux == PORTMUX_TWI0_ALT2_gc) { + port = &PORTC; + } else { + port = &PORTA; + } + #if !defined(__AVR_DA__) && !defined(__AVR_DB__) //DD and EA, and presumably later parts, have an extra mux option. + if (portmux == 3) { + port->OUTCLR = 0x03; // bits 0 and 1 + port->PIN0CTRL |= PORT_PULLUPEN_bm; + port->PIN1CTRL |= PORT_PULLUPEN_bm; + } else + #endif + { + port->OUTCLR = 0x0C; // bits 2 and 3 + port->PIN2CTRL |= PORT_PULLUPEN_bm; + port->PIN3CTRL |= PORT_PULLUPEN_bm; + } + #if defined(TWI0_DUALCTRL) //Also handle slave pins, if enabled + if (TWI0.DUALCTRL & TWI_ENABLE_bm) { + #if !(defined(__AVR_DA__) || defined(__AVR_DB__)) + if (portmux == PORTMUX_TWI0_DEFAULT_gc || + portmux == PORTMUX_TWI0_ALT3_gc) { + PORTC.OUTCLR = 0x0C; + PORTC.PIN2CTRL |= PORT_PULLUPEN_bm; + PORTC.PIN3CTRL |= PORT_PULLUPEN_bm; + } + #else + if (portmux == PORTMUX_TWI0_DEFAULT_gc) { + PORTC.OUTCLR = 0x0C; // bits 2 and 3 + PORTC.PIN2CTRL |= PORT_PULLUPEN_bm; + PORTC.PIN3CTRL |= PORT_PULLUPEN_bm; + } else { + PORTC.OUTCLR = 0xC0; // bits 6 and 7 + PORTC.PIN6CTRL |= PORT_PULLUPEN_bm; + PORTC.PIN7CTRL |= PORT_PULLUPEN_bm; + } + #endif + } + #endif + #elif defined(PORTMUX_TWISPIROUTEA) // megaAVR 0-series + uint8_t portmux = PORTMUX.TWISPIROUTEA & PORTMUX_TWI0_gm; + PORT_t *port; + // Note that all megaAVR 0-series parts have a PORTA, PORTC and PORTF! + if(portmux == PORTMUX_TWI0_ALT2_gc) { + port = &PORTC; + } else { + port = &PORTA; + } + port->OUTCLR = 0x0C; // bits 2 and 3 + port->PIN2CTRL |= PORT_PULLUPEN_bm; + port->PIN3CTRL |= PORT_PULLUPEN_bm; + #if defined(TWI0_DUALCTRL) //Also handle slave pins, if enabled + if (TWI0.DUALCTRL & TWI_ENABLE_bm) { + if (portmux == PORTMUX_TWI0_DEFAULT_gc) { + port = &PORTC; + } else { + port = &PORTF; + } + port->OUTCLR = 0x0C; // bits 2 and 3 + port->PIN2CTRL |= PORT_PULLUPEN_bm; + port->PIN3CTRL |= PORT_PULLUPEN_bm; + } + #endif + #elif defined(MEGATINYCORE) /* tinyAVR 0/1-series */ + #if defined(PORTMUX_TWI0_bm) // 1-series with remappable TWI + if (PORTMUX.CTRLB & PORTMUX_TWI0_bm) { + PORTA.OUTCLR = 0x06; + PORTA.PIN2CTRL |= PORT_PULLUPEN_bm; + PORTA.PIN1CTRL |= PORT_PULLUPEN_bm; + } else { + PORTB.OUTCLR = 0x03; // bits 1 and 0. + PORTB.PIN1CTRL |= PORT_PULLUPEN_bm; + PORTB.PIN0CTRL |= PORT_PULLUPEN_bm; + } + #elif defined(__AVR_ATtinyxy2__) // 8-pin 0/1-series part + PORTA.OUTCLR = 0x06; // bits 2 and 1. + PORTA.PIN2CTRL |= PORT_PULLUPEN_bm; + PORTA.PIN1CTRL |= PORT_PULLUPEN_bm; + #else // 0-series or 2-series part with no remapping options + PORTB.OUTCLR = 0x03; // bits 1 and 0. + PORTB.PIN1CTRL |= PORT_PULLUPEN_bm; + PORTB.PIN0CTRL |= PORT_PULLUPEN_bm; + #endif + #endif +} + +//Check if TWI0 Master pins have a HIGH level: Bit0 = SDA, Bit 1 = SCL +uint8_t TWI0_checkPinLevel(void) { + #if defined(PORTMUX_TWIROUTEA) /* Dx-series */ + uint8_t portmux = (PORTMUX.TWIROUTEA & PORTMUX_TWI0_gm); + VPORT_t *vport; + if (portmux == PORTMUX_TWI0_ALT2_gc) { + vport = &VPORTC; + } else { + vport = &VPORTA; + } + + #if !defined(__AVR_DA__) && !defined(__AVR_DB__) //DD and EA, and presumably later parts, have an extra mux option. + if (portmux == 3) { + return (vport->IN & 0x03); + } else + #endif + { + return ((vport->IN & 0x0C) >> 2); + } + #elif defined(PORTMUX_TWISPIROUTEA) + uint8_t portmux = (PORTMUX.TWISPIROUTEA & PORTMUX_TWI0_gm); + VPORT_t *vport; + if (portmux == PORTMUX_TWI0_ALT2_gc) { + vport = &VPORTC; + } else { + vport = &VPORTA; + } + return ((vport->IN & 0x0C) >> 2); + #elif defined(MEGATINYCORE) /* tinyAVR 0/1-series */ + #if defined(PORTMUX_TWI0_bm) // Has a pin multiplexer + if (PORTMUX.CTRLB & PORTMUX_TWI0_bm) { + return ((VPORTA.IN & 0x06) >> 1); + } else { + return (VPORTB.IN & 0x03); + } + #elif defined(__AVR_ATtinyxy2__) // No PORTMUX for 8-pin parts, they always use PA1/2 + return ((VPORTA.IN & 0x06) >> 1); + #else //it uses PB0/1 + return (VPORTB.IN & 0x03); + #endif + #else + #error "Only modern AVR parts are supported by this version of Wire: tinyAVR 0/1/2-series, megaAVR 0-series, AVR Dx-series or AVR Ex-series." + return 0; + #endif +} + +#if defined(TWI0_DUALCTRL) // full version for parts with dual mode and likely input level too + uint8_t TWI0_setConfig(bool smbuslvl, bool longsetup, uint8_t sda_hold, bool smbuslvl_dual, uint8_t sda_hold_dual) { + uint8_t cfg = TWI0.CTRLA & 0x03; + sda_hold <<= 2; // get these into the right place in the byte + + if (smbuslvl) { + cfg |= 0x40; + } + if (longsetup) { + cfg |= 0x10; + } + cfg |= sda_hold; + TWI0.CTRLA = cfg; + #if defined(TWI0_DUALCTRL) + sda_hold_dual <<= 2; + uint8_t cfg_dual = TWI0.DUALCTRL & 0x03; + if (cfg_dual & 1) { + cfg_dual |= sda_hold_dual; + if (smbuslvl_dual) { + cfg_dual |= 0x40; + } + TWI0.DUALCTRL = cfg_dual; + } else if (sda_hold_dual || smbuslvl_dual) { + return 0x04; // dual mode exists on this part, but is not enabled. This signifies a failure to follow documented startup order + } + #endif + return 0; // return success - all other errors are checked for before this is called. + } +#else // very little to do here on tiny, do save flash with a smaller implementation. + uint8_t TWI0_setConfig(bool longsetup, uint8_t sda_hold) { + uint8_t cfg = TWI0.CTRLA & 0x03; + if (longsetup) { + cfg |= 0x10; + } + cfg |= sda_hold; + TWI0.CTRLA = cfg; + return 0; // return success - all other errors are checked for before this is called. + } +#endif + + +#if defined(TWI1) +void TWI1_ClearPins() { + #if defined(PIN_WIRE1_SDA_PINSWAP_2) || defined(TWI1_DUALCTRL) + uint8_t portmux = PORTMUX.TWIROUTEA & PORTMUX_TWI1_gm; + #endif + #if defined(PIN_WIRE1_SDA_PINSWAP_2) + if (portmux == PORTMUX_TWI1_ALT2_gc) { // make sure we don't get errata'ed + #if defined(PORTB) + PORTB.OUTCLR = 0x0C; // bits 2 and 3 + #endif + } else + #endif + { + PORTF.OUTCLR = 0x0C; // bits 2 and 3 + } + #if defined(TWI1_DUALCTRL) + #if defined(PORTB) + if (TWI1.DUALCTRL & TWI_ENABLE_bm) { + if (portmux == PORTMUX_TWI1_DEFAULT_gc) { + PORTB.OUTCLR = 0x0C; // bits 2 and 3 + } else { + PORTB.OUTCLR = 0xC0; // bits 6 and 7 + } + } + #endif + #endif + (void) portmux; //this is grabbed early, but depending on which part and hence what is conditionally compiled, may not go into the code. It will produce spurious warnings without this line +} + + +bool TWI1_Pins(uint8_t sda_pin, uint8_t scl_pin) { + #if (defined(PIN_WIRE1_SDA_PINSWAP_1) || defined(PIN_WIRE1_SDA_PINSWAP_2)) + // Danger: 'portmux' in this context means all the other settings in portmux, since we're replacing the PORTMUX setting for TWI1, and will bitwise-or with the _gc constants. + // Elsewhere, 'portmux' refers to the setting for this peripheral only, and we compare it to PORTMUX_TWI1_xxx_gc + uint8_t portmux = PORTMUX.TWIROUTEA & ~PORTMUX_TWI1_gm; + #if defined(PIN_WIRE1_SDA_PINSWAP_2) + if (sda_pin == PIN_WIRE1_SDA_PINSWAP_2 && scl_pin == PIN_WIRE1_SCL_PINSWAP_2) { + // Use pin swap + PORTMUX.TWIROUTEA = portmux | PORTMUX_TWI1_ALT2_gc; + return true; + } else + #endif + #if defined(PIN_WIRE1_SDA_PINSWAP_1) + if (sda_pin == PIN_WIRE1_SDA_PINSWAP_1 && scl_pin == PIN_WIRE1_SCL_PINSWAP_1) { + PORTMUX.TWIROUTEA = portmux | PORTMUX_TWI1_ALT1_gc; + return true; + } else + #endif + #if defined(PIN_WIRE1_SDA) + if (sda_pin == PIN_WIRE1_SDA && scl_pin == PIN_WIRE1_SCL) { + // Use default configuration + PORTMUX.TWIROUTEA = portmux; + return true; + } else { + // Assume default configuration + PORTMUX.TWIROUTEA = portmux; + return false; + } + #endif + #else // No TWI1 pin options - why call this? + if (__builtin_constant_p(sda_pin) && __builtin_constant_p(scl_pin)) { + /* constant case - error if there's no swap and the swap attempt is known at compile time */ + if (sda_pin != PIN_WIRE1_SDA || scl_pin != PIN_WIRE1_SCL) { + badCall("This part does not support alternate Wire1 pins, if Wire1.pins() is called, it must be passed the default pins"); + return false; + } else { + return true; + } + } else { /* Non-constant case */ + return (sda_pin == PIN_WIRE1_SDA && scl_pin == PIN_WIRE1_SCL); + } + #endif + return false; +} + + + +bool TWI1_swap(uint8_t state) { + // Danger: 'portmux' in this context means all the other settings in portmux, since we're replacing the PORTMUX setting for TWI1, and will bitwise-or with the _gc constants. + // Elsewhere, 'portmux' refers to the setting for this peripheral only, and we compare it to PORTMUX_TWI1_xxx_gc + uint8_t portmux = PORTMUX.TWIROUTEA & (~PORTMUX_TWI1_gm); + #if defined(PIN_WIRE1_SDA_PINSWAP_2) + if (state == 2) { + // Use pin swap + PORTMUX.TWIROUTEA = portmux | PORTMUX_TWI1_ALT2_gc; + return true; + } else + #endif + #if defined(PIN_WIRE1_SDA_PINSWAP_1) + if (state == 1) { + // Use pin swap + PORTMUX.TWIROUTEA = portmux | PORTMUX_TWI1_ALT1_gc; + return true; + } else + #endif + #if defined(PIN_WIRE1_SDA) + if (state == 0) { + // Use default configuration + PORTMUX.TWIROUTEA = portmux; + return true; + } else { + // Assume default configuration + PORTMUX.TWIROUTEA = portmux; + return false; + } + #else + { + // Assume default configuration + PORTMUX.TWIROUTEA = portmux | PORTMUX_TWI1_ALT2_gc; + return false; + } + #endif + return false; +} + + +void TWI1_usePullups() { + #if defined(PIN_WIRE1_SDA_PINSWAP_2) || defined(TWI1_DUALCTRL) + uint8_t portmux = PORTMUX.TWIROUTEA & PORTMUX_TWI1_gm; + #endif + PORT_t *port; + #if defined(PORTB) //All parts with a TWI1 have a PORTF + if (portmux == PORTMUX_TWI1_ALT2_gc) { + port = &PORTB; + } else { + port = &PORTF; + } + #else + port = &PORTF; + #endif + + port->OUTCLR = 0x0C; // bits 2 and 3 + port->PIN2CTRL |= PORT_PULLUPEN_bm; + port->PIN3CTRL |= PORT_PULLUPEN_bm; + + #if defined(TWI0_DUALCTRL) && defined(PORTB) + if (TWI1.DUALCTRL & TWI_ENABLE_bm) { + if (portmux == PORTMUX_TWI1_DEFAULT_gc) { + PORTB.OUTCLR = 0x0C; // bits 2 and 3 + PORTB.PIN2CTRL |= PORT_PULLUPEN_bm; + PORTB.PIN3CTRL |= PORT_PULLUPEN_bm; + } else { + PORTB.OUTCLR = 0xC0; // bits 6 and 7 + PORTB.PIN6CTRL |= PORT_PULLUPEN_bm; + PORTB.PIN7CTRL |= PORT_PULLUPEN_bm; + } + } + #endif + (void) portmux; //this is grabbed early, but depending on which part and hence what is conditionally compiled, may not go into the code. It will produce spurious warnings without this line +} + +//Check if TWI1 Master pins have a HIGH level: Bit0 = SDA, Bit 1 = SCL +uint8_t TWI1_checkPinLevel(void) { + // we do it this way because when accessed using a pointer, it's no longer any faster to use VPORTx + #if defined(PORTB) //All parts with a TWI1 have a PORTF + uint8_t portmux = PORTMUX.TWIROUTEA & PORTMUX_TWI1_gm; + if (portmux == PORTMUX_TWI1_ALT2_gc) { + return ((VPORTB.IN & 0x0C) >> 2); + } else { + return ((VPORTF.IN & 0x0C) >> 2); + } + #else + return ((VPORTF.IN & 0x0C) >> 2); + #endif +} + +// All devices with TWI1 have dual mode and the most have the smbus levels; the exceptions are caught before this is called +uint8_t TWI1_setConfig(bool smbuslvl, bool longsetup, uint8_t sda_hold, bool smbuslvl_dual, uint8_t sda_hold_dual) { + uint8_t cfg = TWI1.CTRLA & 0x03; + if (smbuslvl) { + cfg |= 0x40; + } + if (longsetup) { + cfg |= 0x10; + } + cfg |= sda_hold; + TWI1.CTRLA = cfg; + #if defined(TWI1_DUALCTRL) + uint8_t cfg_dual = TWI1.DUALCTRL & 0x03; + if (cfg_dual & 1) { + cfg_dual |= sda_hold_dual; + if (smbuslvl_dual) { + cfg_dual = 0x40; + } + TWI1.DUALCTRL = cfg_dual; + } else if (sda_hold_dual || smbuslvl_dual) { + return 0x04; + } + #endif + return 0; +} + + + +#endif /* defined(TWI1) */ + +#endif /* TWI_PINS_H */ \ No newline at end of file