#include "PromAddressDriver.h" #define ADDR_CLK_HI A3 #define ADDR_CLK_LO A4 #define ADDR_DATA A5 void PromAddressDriver::begin() { // The address control pins are always outputs. pinMode(ADDR_DATA, OUTPUT); pinMode(ADDR_CLK_LO, OUTPUT); pinMode(ADDR_CLK_HI, OUTPUT); digitalWrite(ADDR_DATA, LOW); digitalWrite(ADDR_CLK_LO, LOW); digitalWrite(ADDR_CLK_HI, LOW); DDRB |= 0x1c; // Set D10..D12 as outputs // To save time, the setAddress only writes the hi byte if it has changed. // The value used to detect the change is initialized to a non-zero value, // so set an initial address to avoid the the case where the first address // written is the 'magic' initial address. setAddress(0x0000); } // Set a 16 bit address in the two address shift registers and // the upper bits on the extened address pins. void PromAddressDriver::setAddress(uint32_t address) { static byte lastHi = 0xca; static byte lastUpper = 0xca; byte upper = (address >> 16) & 0xff; byte hi = (address >> 8) & 0xff; byte lo = address & 0xff; if (upper != lastUpper) { setUpperAddress(upper); lastUpper = upper; } if (hi != lastHi) { setAddressRegister(ADDR_CLK_HI, hi); lastHi = hi; } setAddressRegister(ADDR_CLK_LO, lo); } void PromAddressDriver::setUpperAddress(byte addr) { // Set the upper address on pins D10..D12. PORTB = (PORTB & 0xe3) | ((addr << 2) & 0x1c); } // Shift an 8-bit value into one of the address shift registers. Note that // the data pins are tied together, selecting the high or low address register // is a matter of using the correct clock pin to shift the data in. void PromAddressDriver::setAddressRegister(uint8_t clkPin, byte addr) { byte mask = 0; if (clkPin == ADDR_CLK_HI) mask = 0x08; else if (clkPin == ADDR_CLK_LO) mask = 0x10; // Make sure the clock is low to start. PORTC &= ~mask; // Shift 8 bits in, starting with the MSB. for (int ix = 0; (ix < 8); ix++) { // Set the data bit if (addr & 0x80) { PORTC |= 0x20; } else { PORTC &= 0xdf; } // Toggle the clock high then low PORTC |= mask; delayMicroseconds(3); PORTC &= ~mask; addr <<= 1; } }