Add support for Meira
This commit is contained in:
		
							parent
							
								
									c51dfef958
								
							
						
					
					
						commit
						a9a46adba0
					
				
					 23 changed files with 2186 additions and 0 deletions
				
			
		
							
								
								
									
										287
									
								
								keyboards/meira/TWIlib.c
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										287
									
								
								keyboards/meira/TWIlib.c
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,287 @@ | ||||||
|  | /*
 | ||||||
|  |  * TWIlib.c | ||||||
|  |  * | ||||||
|  |  *  Created: 6/01/2014 10:41:33 PM | ||||||
|  |  *  Author: Chris Herring | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <avr/io.h> | ||||||
|  | #include <avr/interrupt.h> | ||||||
|  | #include "TWIlib.h" | ||||||
|  | #include "util/delay.h" | ||||||
|  | #include "print.h" | ||||||
|  | 
 | ||||||
|  | void TWIInit() | ||||||
|  | { | ||||||
|  | 	TWIInfo.mode = Ready; | ||||||
|  | 	TWIInfo.errorCode = 0xFF; | ||||||
|  | 	TWIInfo.repStart = 0; | ||||||
|  | 	// Set pre-scalers (no pre-scaling)
 | ||||||
|  | 	TWSR = 0; | ||||||
|  | 	// Set bit rate
 | ||||||
|  | 	TWBR = ((F_CPU / TWI_FREQ) - 16) / 2; | ||||||
|  | 	// Enable TWI and interrupt
 | ||||||
|  | 	TWCR = (1 << TWIE) | (1 << TWEN); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint8_t isTWIReady() | ||||||
|  | { | ||||||
|  | 	if ( (TWIInfo.mode == Ready) | (TWIInfo.mode == RepeatedStartSent) ) | ||||||
|  | 	{ | ||||||
|  | 
 | ||||||
|  | //        xprintf("i2c ready\n");
 | ||||||
|  | 		return 1; | ||||||
|  | 	} | ||||||
|  | 	else | ||||||
|  | 	{ | ||||||
|  | 		if(TWIInfo.mode == Initializing){ | ||||||
|  | 			switch(TWIInfo.errorCode){ | ||||||
|  | 		        case TWI_SUCCESS: | ||||||
|  | 		            break; | ||||||
|  | 		        case TWI_NO_RELEVANT_INFO: | ||||||
|  | 
 | ||||||
|  | 		        	break; | ||||||
|  | 				case TWI_LOST_ARBIT: | ||||||
|  | 				case TWI_MT_DATA_NACK: | ||||||
|  | 					// Some kind of I2C error, reset and re-init
 | ||||||
|  | 		        	xprintf("I2C init error: %d\n", TWIInfo.errorCode); | ||||||
|  | 			        TWCR = (1 << TWINT)|(1 << TWSTO); | ||||||
|  | 		        	TWIInit(); | ||||||
|  | 		        	break; | ||||||
|  | 		        default: | ||||||
|  | 		        	xprintf("Other i2c init error: %d\n", TWIInfo.errorCode); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void TWITransmitData(void *const TXdata, uint8_t dataLen, uint8_t repStart, uint8_t blocking) | ||||||
|  | { | ||||||
|  | 	// Wait until ready
 | ||||||
|  | 	while (!isTWIReady()) {_delay_us(1);} | ||||||
|  | 	// Reset the I2C stuff
 | ||||||
|  | 	TWCR = (1 << TWINT)|(1 << TWSTO); | ||||||
|  | 	TWIInit(); | ||||||
|  | 	// Set repeated start mode
 | ||||||
|  | 	TWIInfo.repStart = repStart; | ||||||
|  | 	// Copy transmit info to global variables
 | ||||||
|  | 	TWITransmitBuffer = (uint8_t *)TXdata; | ||||||
|  | 	TXBuffLen = dataLen; | ||||||
|  | 	TXBuffIndex = 0; | ||||||
|  | 
 | ||||||
|  | 	// If a repeated start has been sent, then devices are already listening for an address
 | ||||||
|  | 	// and another start does not need to be sent.
 | ||||||
|  | 	if (TWIInfo.mode == RepeatedStartSent) | ||||||
|  | 	{ | ||||||
|  | 		TWIInfo.mode = Initializing; | ||||||
|  | 		TWDR = TWITransmitBuffer[TXBuffIndex++]; // Load data to transmit buffer
 | ||||||
|  | 		TWISendTransmit(); // Send the data
 | ||||||
|  | 	} | ||||||
|  | 	else // Otherwise, just send the normal start signal to begin transmission.
 | ||||||
|  | 	{ | ||||||
|  | 		TWIInfo.mode = Initializing; | ||||||
|  | 		TWISendStart(); | ||||||
|  | 	} | ||||||
|  | 	if(blocking){ | ||||||
|  | 		// Wait until ready
 | ||||||
|  | 		while (!isTWIReady()){_delay_us(1);} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // uint8_t TWITransmitData(void *const TXdata, uint8_t dataLen, uint8_t repStart)
 | ||||||
|  | // {
 | ||||||
|  | // 	if (dataLen <= TXMAXBUFLEN)
 | ||||||
|  | // 	{
 | ||||||
|  | // 		// Wait until ready
 | ||||||
|  | // 		while (!isTWIReady()) {_delay_us(1);}
 | ||||||
|  | // 		// Set repeated start mode
 | ||||||
|  | // 		TWIInfo.repStart = repStart;
 | ||||||
|  | // 		// Copy data into the transmit buffer
 | ||||||
|  | // 		uint8_t *data = (uint8_t *)TXdata;
 | ||||||
|  | // 		for (int i = 0; i < dataLen; i++)
 | ||||||
|  | // 		{
 | ||||||
|  | // 			TWITransmitBuffer[i] = data[i];
 | ||||||
|  | // 		}
 | ||||||
|  | // 		// Copy transmit info to global variables
 | ||||||
|  | // 		TXBuffLen = dataLen;
 | ||||||
|  | // 		TXBuffIndex = 0;
 | ||||||
|  | 
 | ||||||
|  | // 		// If a repeated start has been sent, then devices are already listening for an address
 | ||||||
|  | // 		// and another start does not need to be sent.
 | ||||||
|  | // 		if (TWIInfo.mode == RepeatedStartSent)
 | ||||||
|  | // 		{
 | ||||||
|  | // 			TWIInfo.mode = Initializing;
 | ||||||
|  | // 			TWDR = TWITransmitBuffer[TXBuffIndex++]; // Load data to transmit buffer
 | ||||||
|  | // 			TWISendTransmit(); // Send the data
 | ||||||
|  | // 		}
 | ||||||
|  | // 		else // Otherwise, just send the normal start signal to begin transmission.
 | ||||||
|  | // 		{
 | ||||||
|  | // 			TWIInfo.mode = Initializing;
 | ||||||
|  | // 			TWISendStart();
 | ||||||
|  | // 		}
 | ||||||
|  | 
 | ||||||
|  | // 	}
 | ||||||
|  | // 	else
 | ||||||
|  | // 	{
 | ||||||
|  | // 		return 1; // return an error if data length is longer than buffer
 | ||||||
|  | // 	}
 | ||||||
|  | // 	return 0;
 | ||||||
|  | // }
 | ||||||
|  | 
 | ||||||
|  | uint8_t TWIReadData(uint8_t TWIaddr, uint8_t bytesToRead, uint8_t repStart) | ||||||
|  | { | ||||||
|  | 	// Check if number of bytes to read can fit in the RXbuffer
 | ||||||
|  | 	if (bytesToRead < RXMAXBUFLEN) | ||||||
|  | 	{ | ||||||
|  | 		// Reset buffer index and set RXBuffLen to the number of bytes to read
 | ||||||
|  | 		RXBuffIndex = 0; | ||||||
|  | 		RXBuffLen = bytesToRead; | ||||||
|  | 		// Create the one value array for the address to be transmitted
 | ||||||
|  | 		uint8_t TXdata[1]; | ||||||
|  | 		// Shift the address and AND a 1 into the read write bit (set to write mode)
 | ||||||
|  | 		TXdata[0] = (TWIaddr << 1) | 0x01; | ||||||
|  | 		// Use the TWITransmitData function to initialize the transfer and address the slave
 | ||||||
|  | 		TWITransmitData(TXdata, 1, repStart, 0); | ||||||
|  | 	} | ||||||
|  | 	else | ||||||
|  | 	{ | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | ISR (TWI_vect) | ||||||
|  | { | ||||||
|  | 	switch (TWI_STATUS) | ||||||
|  | 	{ | ||||||
|  | 		// ----\/ ---- MASTER TRANSMITTER OR WRITING ADDRESS ----\/ ----  //
 | ||||||
|  | 		case TWI_MT_SLAW_ACK: // SLA+W transmitted and ACK received
 | ||||||
|  | 		// Set mode to Master Transmitter
 | ||||||
|  | 		TWIInfo.mode = MasterTransmitter; | ||||||
|  | 		case TWI_START_SENT: // Start condition has been transmitted
 | ||||||
|  | 		case TWI_MT_DATA_ACK: // Data byte has been transmitted, ACK received
 | ||||||
|  | 			if (TXBuffIndex < TXBuffLen) // If there is more data to send
 | ||||||
|  | 			{ | ||||||
|  | 				TWDR = TWITransmitBuffer[TXBuffIndex++]; // Load data to transmit buffer
 | ||||||
|  | 				TWIInfo.errorCode = TWI_NO_RELEVANT_INFO; | ||||||
|  | 				TWISendTransmit(); // Send the data
 | ||||||
|  | 			} | ||||||
|  | 			// This transmission is complete however do not release bus yet
 | ||||||
|  | 			else if (TWIInfo.repStart) | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.errorCode = 0xFF; | ||||||
|  | 				TWISendStart(); | ||||||
|  | 			} | ||||||
|  | 			// All transmissions are complete, exit
 | ||||||
|  | 			else | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.mode = Ready; | ||||||
|  | 				TWIInfo.errorCode = 0xFF; | ||||||
|  | 				TWISendStop(); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 		// ----\/ ---- MASTER RECEIVER ----\/ ----  //
 | ||||||
|  | 
 | ||||||
|  | 		case TWI_MR_SLAR_ACK: // SLA+R has been transmitted, ACK has been received
 | ||||||
|  | 			// Switch to Master Receiver mode
 | ||||||
|  | 			TWIInfo.mode = MasterReceiver; | ||||||
|  | 			// If there is more than one byte to be read, receive data byte and return an ACK
 | ||||||
|  | 			if (RXBuffIndex < RXBuffLen-1) | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.errorCode = TWI_NO_RELEVANT_INFO; | ||||||
|  | 				TWISendACK(); | ||||||
|  | 			} | ||||||
|  | 			// Otherwise when a data byte (the only data byte) is received, return NACK
 | ||||||
|  | 			else | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.errorCode = TWI_NO_RELEVANT_INFO; | ||||||
|  | 				TWISendNACK(); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 		case TWI_MR_DATA_ACK: // Data has been received, ACK has been transmitted.
 | ||||||
|  | 
 | ||||||
|  | 			/// -- HANDLE DATA BYTE --- ///
 | ||||||
|  | 			TWIReceiveBuffer[RXBuffIndex++] = TWDR; | ||||||
|  | 			// If there is more than one byte to be read, receive data byte and return an ACK
 | ||||||
|  | 			if (RXBuffIndex < RXBuffLen-1) | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.errorCode = TWI_NO_RELEVANT_INFO; | ||||||
|  | 				TWISendACK(); | ||||||
|  | 			} | ||||||
|  | 			// Otherwise when a data byte (the only data byte) is received, return NACK
 | ||||||
|  | 			else | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.errorCode = TWI_NO_RELEVANT_INFO; | ||||||
|  | 				TWISendNACK(); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 		case TWI_MR_DATA_NACK: // Data byte has been received, NACK has been transmitted. End of transmission.
 | ||||||
|  | 
 | ||||||
|  | 			/// -- HANDLE DATA BYTE --- ///
 | ||||||
|  | 			TWIReceiveBuffer[RXBuffIndex++] = TWDR; | ||||||
|  | 			// This transmission is complete however do not release bus yet
 | ||||||
|  | 			if (TWIInfo.repStart) | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.errorCode = 0xFF; | ||||||
|  | 				TWISendStart(); | ||||||
|  | 			} | ||||||
|  | 			// All transmissions are complete, exit
 | ||||||
|  | 			else | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.mode = Ready; | ||||||
|  | 				TWIInfo.errorCode = 0xFF; | ||||||
|  | 				TWISendStop(); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 		// ----\/ ---- MT and MR common ----\/ ---- //
 | ||||||
|  | 
 | ||||||
|  | 		case TWI_MR_SLAR_NACK: // SLA+R transmitted, NACK received
 | ||||||
|  | 		case TWI_MT_SLAW_NACK: // SLA+W transmitted, NACK received
 | ||||||
|  | 		case TWI_MT_DATA_NACK: // Data byte has been transmitted, NACK received
 | ||||||
|  | 		case TWI_LOST_ARBIT: // Arbitration has been lost
 | ||||||
|  | 			// Return error and send stop and set mode to ready
 | ||||||
|  | 			if (TWIInfo.repStart) | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.errorCode = TWI_STATUS; | ||||||
|  | 				TWISendStart(); | ||||||
|  | 			} | ||||||
|  | 			// All transmissions are complete, exit
 | ||||||
|  | 			else | ||||||
|  | 			{ | ||||||
|  | 				TWIInfo.mode = Ready; | ||||||
|  | 				TWIInfo.errorCode = TWI_STATUS; | ||||||
|  | 				TWISendStop(); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case TWI_REP_START_SENT: // Repeated start has been transmitted
 | ||||||
|  | 			// Set the mode but DO NOT clear TWINT as the next data is not yet ready
 | ||||||
|  | 			TWIInfo.mode = RepeatedStartSent; | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 		// ----\/ ---- SLAVE RECEIVER ----\/ ----  //
 | ||||||
|  | 
 | ||||||
|  | 		// TODO  IMPLEMENT SLAVE RECEIVER FUNCTIONALITY
 | ||||||
|  | 
 | ||||||
|  | 		// ----\/ ---- SLAVE TRANSMITTER ----\/ ----  //
 | ||||||
|  | 
 | ||||||
|  | 		// TODO  IMPLEMENT SLAVE TRANSMITTER FUNCTIONALITY
 | ||||||
|  | 
 | ||||||
|  | 		// ----\/ ---- MISCELLANEOUS STATES ----\/ ----  //
 | ||||||
|  | 		case TWI_NO_RELEVANT_INFO: // It is not really possible to get into this ISR on this condition
 | ||||||
|  | 								   // Rather, it is there to be manually set between operations
 | ||||||
|  | 			break; | ||||||
|  | 		case TWI_ILLEGAL_START_STOP: // Illegal START/STOP, abort and return error
 | ||||||
|  | 			TWIInfo.errorCode = TWI_ILLEGAL_START_STOP; | ||||||
|  | 			TWIInfo.mode = Ready; | ||||||
|  | 			TWISendStop(); | ||||||
|  | 			break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										81
									
								
								keyboards/meira/TWIlib.h
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										81
									
								
								keyboards/meira/TWIlib.h
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,81 @@ | ||||||
|  | /*
 | ||||||
|  |  * TWIlib.h | ||||||
|  |  * | ||||||
|  |  * Created: 6/01/2014 10:38:42 PM | ||||||
|  |  *  Author: Chris Herring | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #ifndef TWILIB_H_ | ||||||
|  | #define TWILIB_H_ | ||||||
|  | // TWI bit rate
 | ||||||
|  | #define TWI_FREQ 400000 | ||||||
|  | // Get TWI status
 | ||||||
|  | #define TWI_STATUS	(TWSR & 0xF8) | ||||||
|  | // Transmit buffer length
 | ||||||
|  | #define TXMAXBUFLEN 20 | ||||||
|  | // Receive buffer length
 | ||||||
|  | #define RXMAXBUFLEN 20 | ||||||
|  | // Global transmit buffer
 | ||||||
|  | volatile uint8_t *TWITransmitBuffer; | ||||||
|  | // Global receive buffer
 | ||||||
|  | volatile uint8_t TWIReceiveBuffer[RXMAXBUFLEN]; | ||||||
|  | // Buffer indexes
 | ||||||
|  | volatile int TXBuffIndex; // Index of the transmit buffer. Is volatile, can change at any time.
 | ||||||
|  | int RXBuffIndex; // Current index in the receive buffer
 | ||||||
|  | // Buffer lengths
 | ||||||
|  | int TXBuffLen; // The total length of the transmit buffer
 | ||||||
|  | int RXBuffLen; // The total number of bytes to read (should be less than RXMAXBUFFLEN)
 | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  | 	Ready, | ||||||
|  | 	Initializing, | ||||||
|  | 	RepeatedStartSent, | ||||||
|  | 	MasterTransmitter, | ||||||
|  | 	MasterReceiver, | ||||||
|  | 	SlaceTransmitter, | ||||||
|  | 	SlaveReciever | ||||||
|  | 	} TWIMode; | ||||||
|  | 
 | ||||||
|  |  typedef struct TWIInfoStruct{ | ||||||
|  | 	TWIMode mode; | ||||||
|  | 	uint8_t errorCode; | ||||||
|  | 	uint8_t repStart; | ||||||
|  | 	}TWIInfoStruct; | ||||||
|  | TWIInfoStruct TWIInfo; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // TWI Status Codes
 | ||||||
|  | #define TWI_START_SENT			0x08 // Start sent
 | ||||||
|  | #define TWI_REP_START_SENT		0x10 // Repeated Start sent
 | ||||||
|  | // Master Transmitter Mode
 | ||||||
|  | #define TWI_MT_SLAW_ACK			0x18 // SLA+W sent and ACK received
 | ||||||
|  | #define TWI_MT_SLAW_NACK		0x20 // SLA+W sent and NACK received
 | ||||||
|  | #define TWI_MT_DATA_ACK			0x28 // DATA sent and ACK received
 | ||||||
|  | #define TWI_MT_DATA_NACK		0x30 // DATA sent and NACK received
 | ||||||
|  | // Master Receiver Mode
 | ||||||
|  | #define TWI_MR_SLAR_ACK			0x40 // SLA+R sent, ACK received
 | ||||||
|  | #define TWI_MR_SLAR_NACK		0x48 // SLA+R sent, NACK received
 | ||||||
|  | #define TWI_MR_DATA_ACK			0x50 // Data received, ACK returned
 | ||||||
|  | #define TWI_MR_DATA_NACK		0x58 // Data received, NACK returned
 | ||||||
|  | 
 | ||||||
|  | // Miscellaneous States
 | ||||||
|  | #define TWI_LOST_ARBIT			0x38 // Arbitration has been lost
 | ||||||
|  | #define TWI_NO_RELEVANT_INFO	0xF8 // No relevant information available
 | ||||||
|  | #define TWI_ILLEGAL_START_STOP	0x00 // Illegal START or STOP condition has been detected
 | ||||||
|  | #define TWI_SUCCESS				0xFF // Successful transfer, this state is impossible from TWSR as bit2 is 0 and read only
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #define TWISendStart()		(TWCR = (1<<TWINT)|(1<<TWSTA)|(1<<TWEN)|(1<<TWIE)) // Send the START signal, enable interrupts and TWI, clear TWINT flag to resume transfer.
 | ||||||
|  | #define TWISendStop()		(TWCR = (1<<TWINT)|(1<<TWSTO)|(1<<TWEN)|(1<<TWIE)) // Send the STOP signal, enable interrupts and TWI, clear TWINT flag.
 | ||||||
|  | #define TWISendTransmit()	(TWCR = (1<<TWINT)|(1<<TWEN)|(1<<TWIE)) // Used to resume a transfer, clear TWINT and ensure that TWI and interrupts are enabled.
 | ||||||
|  | #define TWISendACK()		(TWCR = (1<<TWINT)|(1<<TWEN)|(1<<TWIE)|(1<<TWEA)) // FOR MR mode. Resume a transfer, ensure that TWI and interrupts are enabled and respond with an ACK if the device is addressed as a slave or after it receives a byte.
 | ||||||
|  | #define TWISendNACK()		(TWCR = (1<<TWINT)|(1<<TWEN)|(1<<TWIE)) // FOR MR mode. Resume a transfer, ensure that TWI and interrupts are enabled but DO NOT respond with an ACK if the device is addressed as a slave or after it receives a byte.
 | ||||||
|  | 
 | ||||||
|  | // Function declarations
 | ||||||
|  | void TWITransmitData(void *const TXdata, uint8_t dataLen, uint8_t repStart, uint8_t blocking); | ||||||
|  | void TWIInit(void); | ||||||
|  | uint8_t TWIReadData(uint8_t TWIaddr, uint8_t bytesToRead, uint8_t repStart); | ||||||
|  | uint8_t isTWIReady(void); | ||||||
|  | 
 | ||||||
|  | #endif // TWICOMMS_H_
 | ||||||
							
								
								
									
										53
									
								
								keyboards/meira/config.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								keyboards/meira/config.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,53 @@ | ||||||
|  | /*
 | ||||||
|  | Copyright 2017 Cole Markham | ||||||
|  | 
 | ||||||
|  | This program is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 2 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | This program is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | You should have received a copy of the GNU General Public License | ||||||
|  | along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | #ifndef CONFIG_H | ||||||
|  | #define CONFIG_H | ||||||
|  | 
 | ||||||
|  | #include "config_common.h" | ||||||
|  | 
 | ||||||
|  | /* USB Device descriptor parameter */ | ||||||
|  | #define VENDOR_ID       0xFEED | ||||||
|  | #define PRODUCT_ID      0x6061 | ||||||
|  | #define DEVICE_VER      0x0001 | ||||||
|  | #define MANUFACTURER    WoodKeys.click | ||||||
|  | #define PRODUCT         Meira | ||||||
|  | #define DESCRIPTION     Low-profile Ortholinear Compact keyboard | ||||||
|  | 
 | ||||||
|  | /* key matrix size */ | ||||||
|  | #define MATRIX_ROWS 4 | ||||||
|  | #define MATRIX_COLS 12 | ||||||
|  | 
 | ||||||
|  | /* COL2ROW, ROW2COL, or CUSTOM_MATRIX */ | ||||||
|  | #define DIODE_DIRECTION CUSTOM_MATRIX | ||||||
|  | 
 | ||||||
|  | #define BACKLIGHT_LEVELS 10 | ||||||
|  | #define BACKLIGHT_PWM_MAP {2, 4, 8, 16, 40, 55, 70, 128, 200, 255} | ||||||
|  | #define BACKLIGHT_BREATHING | ||||||
|  | 
 | ||||||
|  | #define RGB_DI_PIN D3 | ||||||
|  | #define RGBLIGHT_TIMER | ||||||
|  | #define RGBLED_NUM 15    // Number of LEDs
 | ||||||
|  | 
 | ||||||
|  | #ifdef SUBPROJECT_promicro | ||||||
|  |     #include "promicro/config.h" | ||||||
|  | #endif | ||||||
|  | #ifdef SUBPROJECT_featherble | ||||||
|  |     #include "featherble/config.h" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										175
									
								
								keyboards/meira/featherble/config.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										175
									
								
								keyboards/meira/featherble/config.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,175 @@ | ||||||
|  | /*
 | ||||||
|  | Copyright 2017 REPLACE_WITH_YOUR_NAME | ||||||
|  | 
 | ||||||
|  | This program is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 2 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | This program is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | You should have received a copy of the GNU General Public License | ||||||
|  | along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | #ifndef FEATHERBLECONFIG_H | ||||||
|  | #define FEATHERBLECONFIG_H | ||||||
|  | 
 | ||||||
|  | #include "config_common.h" | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Keyboard Matrix Assignments | ||||||
|  |  * | ||||||
|  |  * Change this to how you wired your keyboard | ||||||
|  |  * COLS: AVR pins used for columns, left to right | ||||||
|  |  * ROWS: AVR pins used for rows, top to bottom | ||||||
|  |  * DIODE_DIRECTION: COL2ROW = COL = Anode (+), ROW = Cathode (-, marked on diode) | ||||||
|  |  *                  ROW2COL = ROW = Anode (+), COL = Cathode (-, marked on diode) | ||||||
|  |  * | ||||||
|  | */ | ||||||
|  | #define MATRIX_ROW_PINS { F7, F6, F5, F4 } | ||||||
|  | // Column pins to demux in LSB order
 | ||||||
|  | #define MATRIX_COL_PINS { C7, B7, B6, C6 } | ||||||
|  | #define LED_EN_PIN D2 | ||||||
|  | #define UNUSED_PINS | ||||||
|  | 
 | ||||||
|  | #define CATERINA_BOOTLOADER | ||||||
|  | 
 | ||||||
|  |   | ||||||
|  | // #define BACKLIGHT_PIN B7
 | ||||||
|  | // #define BACKLIGHT_BREATHING
 | ||||||
|  | //#define BACKLIGHT_LEVELS 3
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /* Debounce reduces chatter (unintended double-presses) - set 0 if debouncing is not needed */ | ||||||
|  | #define DEBOUNCING_DELAY 5 | ||||||
|  | 
 | ||||||
|  | /* define if matrix has ghost (lacks anti-ghosting diodes) */ | ||||||
|  | //#define MATRIX_HAS_GHOST
 | ||||||
|  | 
 | ||||||
|  | /* number of backlight levels */ | ||||||
|  | 
 | ||||||
|  | /* Mechanical locking support. Use KC_LCAP, KC_LNUM or KC_LSCR instead in keymap */ | ||||||
|  | #define LOCKING_SUPPORT_ENABLE | ||||||
|  | /* Locking resynchronize hack */ | ||||||
|  | #define LOCKING_RESYNC_ENABLE | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Force NKRO | ||||||
|  |  * | ||||||
|  |  * Force NKRO (nKey Rollover) to be enabled by default, regardless of the saved | ||||||
|  |  * state in the bootmagic EEPROM settings. (Note that NKRO must be enabled in the | ||||||
|  |  * makefile for this to work.) | ||||||
|  |  * | ||||||
|  |  * If forced on, NKRO can be disabled via magic key (default = LShift+RShift+N) | ||||||
|  |  * until the next keyboard reset. | ||||||
|  |  * | ||||||
|  |  * NKRO may prevent your keystrokes from being detected in the BIOS, but it is | ||||||
|  |  * fully operational during normal computer usage. | ||||||
|  |  * | ||||||
|  |  * For a less heavy-handed approach, enable NKRO via magic key (LShift+RShift+N) | ||||||
|  |  * or via bootmagic (hold SPACE+N while plugging in the keyboard). Once set by | ||||||
|  |  * bootmagic, NKRO mode will always be enabled until it is toggled again during a | ||||||
|  |  * power-up. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | //#define FORCE_NKRO
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Magic Key Options | ||||||
|  |  * | ||||||
|  |  * Magic keys are hotkey commands that allow control over firmware functions of | ||||||
|  |  * the keyboard. They are best used in combination with the HID Listen program, | ||||||
|  |  * found here: https://www.pjrc.com/teensy/hid_listen.html
 | ||||||
|  |  * | ||||||
|  |  * The options below allow the magic key functionality to be changed. This is | ||||||
|  |  * useful if your keyboard/keypad is missing keys and you want magic key support. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* key combination for magic key command */ | ||||||
|  | #define IS_COMMAND() ( \ | ||||||
|  |     keyboard_report->mods == (MOD_BIT(KC_LSHIFT) | MOD_BIT(KC_RSHIFT)) \ | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | /* control how magic key switches layers */ | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_FKEYS  true
 | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_NKEYS  true
 | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_CUSTOM false
 | ||||||
|  | 
 | ||||||
|  | /* override magic key keymap */ | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_FKEYS
 | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_NKEYS
 | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_CUSTOM
 | ||||||
|  | //#define MAGIC_KEY_HELP1          H
 | ||||||
|  | //#define MAGIC_KEY_HELP2          SLASH
 | ||||||
|  | //#define MAGIC_KEY_DEBUG          D
 | ||||||
|  | //#define MAGIC_KEY_DEBUG_MATRIX   X
 | ||||||
|  | //#define MAGIC_KEY_DEBUG_KBD      K
 | ||||||
|  | //#define MAGIC_KEY_DEBUG_MOUSE    M
 | ||||||
|  | //#define MAGIC_KEY_VERSION        V
 | ||||||
|  | //#define MAGIC_KEY_STATUS         S
 | ||||||
|  | //#define MAGIC_KEY_CONSOLE        C
 | ||||||
|  | //#define MAGIC_KEY_LAYER0_ALT1    ESC
 | ||||||
|  | //#define MAGIC_KEY_LAYER0_ALT2    GRAVE
 | ||||||
|  | //#define MAGIC_KEY_LAYER0         0
 | ||||||
|  | //#define MAGIC_KEY_LAYER1         1
 | ||||||
|  | //#define MAGIC_KEY_LAYER2         2
 | ||||||
|  | //#define MAGIC_KEY_LAYER3         3
 | ||||||
|  | //#define MAGIC_KEY_LAYER4         4
 | ||||||
|  | //#define MAGIC_KEY_LAYER5         5
 | ||||||
|  | //#define MAGIC_KEY_LAYER6         6
 | ||||||
|  | //#define MAGIC_KEY_LAYER7         7
 | ||||||
|  | //#define MAGIC_KEY_LAYER8         8
 | ||||||
|  | //#define MAGIC_KEY_LAYER9         9
 | ||||||
|  | //#define MAGIC_KEY_BOOTLOADER     PAUSE
 | ||||||
|  | //#define MAGIC_KEY_LOCK           CAPS
 | ||||||
|  | //#define MAGIC_KEY_EEPROM         E
 | ||||||
|  | //#define MAGIC_KEY_NKRO           N
 | ||||||
|  | //#define MAGIC_KEY_SLEEP_LED      Z
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Feature disable options | ||||||
|  |  *  These options are also useful to firmware size reduction. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* disable debug print */ | ||||||
|  | //#define NO_DEBUG
 | ||||||
|  | 
 | ||||||
|  | /* disable print */ | ||||||
|  | //#define NO_PRINT
 | ||||||
|  | 
 | ||||||
|  | /* disable action features */ | ||||||
|  | //#define NO_ACTION_LAYER
 | ||||||
|  | //#define NO_ACTION_TAPPING
 | ||||||
|  | //#define NO_ACTION_ONESHOT
 | ||||||
|  | //#define NO_ACTION_MACRO
 | ||||||
|  | //#define NO_ACTION_FUNCTION
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * MIDI options | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* Prevent use of disabled MIDI features in the keymap */ | ||||||
|  | //#define MIDI_ENABLE_STRICT 1
 | ||||||
|  | 
 | ||||||
|  | /* enable basic MIDI features:
 | ||||||
|  |    - MIDI notes can be sent when in Music mode is on | ||||||
|  | */ | ||||||
|  | //#define MIDI_BASIC
 | ||||||
|  | 
 | ||||||
|  | /* enable advanced MIDI features:
 | ||||||
|  |    - MIDI notes can be added to the keymap | ||||||
|  |    - Octave shift and transpose | ||||||
|  |    - Virtual sustain, portamento, and modulation wheel | ||||||
|  |    - etc. | ||||||
|  | */ | ||||||
|  | //#define MIDI_ADVANCED
 | ||||||
|  | 
 | ||||||
|  | /* override number of MIDI tone keycodes (each octave adds 12 keycodes and allocates 12 bytes) */ | ||||||
|  | //#define MIDI_TONE_KEYCODE_OCTAVES 1
 | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										1
									
								
								keyboards/meira/featherble/featherble.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								keyboards/meira/featherble/featherble.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | ||||||
|  | #include "meira.h" | ||||||
							
								
								
									
										9
									
								
								keyboards/meira/featherble/featherble.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								keyboards/meira/featherble/featherble.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,9 @@ | ||||||
|  | #ifndef FEATHERBLE_H | ||||||
|  | #define FEATHERBLE_H | ||||||
|  | 
 | ||||||
|  | #include "../meira.h" | ||||||
|  | 
 | ||||||
|  | #include "quantum.h" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										4
									
								
								keyboards/meira/featherble/rules.mk
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								keyboards/meira/featherble/rules.mk
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,4 @@ | ||||||
|  | BLUETOOTH_ENABLE = yes | ||||||
|  | BACKLIGHT_ENABLE = yes | ||||||
|  | F_CPU = 8000000 | ||||||
|  | 
 | ||||||
							
								
								
									
										286
									
								
								keyboards/meira/issi.c
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										286
									
								
								keyboards/meira/issi.c
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,286 @@ | ||||||
|  | #ifdef ISSI_ENABLE | ||||||
|  | 
 | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <stdint.h> | ||||||
|  | #include <util/delay.h> | ||||||
|  | #include <avr/sfr_defs.h> | ||||||
|  | #include <avr/io.h> | ||||||
|  | #include <util/twi.h> | ||||||
|  | #include "issi.h" | ||||||
|  | #include "print.h" | ||||||
|  | #include "TWIlib.h" | ||||||
|  | 
 | ||||||
|  | #define ISSI_ADDR_DEFAULT 0xE8 | ||||||
|  | 
 | ||||||
|  | #define ISSI_REG_CONFIG 0x00 | ||||||
|  | #define ISSI_REG_CONFIG_PICTUREMODE 0x00 | ||||||
|  | #define ISSI_REG_CONFIG_AUTOPLAYMODE 0x08 | ||||||
|  | 
 | ||||||
|  | #define ISSI_CONF_PICTUREMODE 0x00 | ||||||
|  | #define ISSI_CONF_AUTOFRAMEMODE 0x04 | ||||||
|  | #define ISSI_CONF_AUDIOMODE 0x08 | ||||||
|  | 
 | ||||||
|  | #define ISSI_REG_PICTUREFRAME 0x01 | ||||||
|  | 
 | ||||||
|  | #define ISSI_REG_SHUTDOWN 0x0A | ||||||
|  | #define ISSI_REG_AUDIOSYNC 0x06 | ||||||
|  | 
 | ||||||
|  | #define ISSI_COMMANDREGISTER 0xFD | ||||||
|  | #define ISSI_BANK_FUNCTIONREG 0x0B // helpfully called 'page nine'
 | ||||||
|  | uint8_t control[8][9] = { | ||||||
|  |     {0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||||||
|  |     {0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||||||
|  |     {0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||||||
|  |     {0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||||||
|  | }; | ||||||
|  | ISSIDeviceStruct *issi_devices[4] = {0, 0, 0, 0}; | ||||||
|  | 
 | ||||||
|  | #ifndef cbi | ||||||
|  | #define cbi(sfr, bit) (_SFR_BYTE(sfr) &= ~_BV(bit)) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifndef sbi | ||||||
|  | #define sbi(sfr, bit) (_SFR_BYTE(sfr) |= _BV(bit)) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #define I2C_WRITE 0 | ||||||
|  | #define F_SCL 400000UL // SCL frequency
 | ||||||
|  | #define Prescaler 1 | ||||||
|  | #define TWBR_val ((((F_CPU / F_SCL) / Prescaler) - 16 ) / 2) | ||||||
|  | 
 | ||||||
|  | uint8_t i2c_start(uint8_t address) | ||||||
|  | { | ||||||
|  |     // reset TWI control register
 | ||||||
|  |     TWCR = 0; | ||||||
|  |     // transmit START condition
 | ||||||
|  |     TWCR = (1<<TWINT) | (1<<TWSTA) | (1<<TWEN); | ||||||
|  |     // wait for end of transmission
 | ||||||
|  |     while( !(TWCR & (1<<TWINT)) ); | ||||||
|  | 
 | ||||||
|  |     // check if the start condition was successfully transmitted
 | ||||||
|  |     if((TWSR & 0xF8) != TW_START){ return 1; } | ||||||
|  | 
 | ||||||
|  |     // load slave address into data register
 | ||||||
|  |     TWDR = address; | ||||||
|  |     // start transmission of address
 | ||||||
|  |     TWCR = (1<<TWINT) | (1<<TWEN); | ||||||
|  |     // wait for end of transmission
 | ||||||
|  |     while( !(TWCR & (1<<TWINT)) ); | ||||||
|  | 
 | ||||||
|  |     // check if the device has acknowledged the READ / WRITE mode
 | ||||||
|  |     uint8_t twst = TW_STATUS & 0xF8; | ||||||
|  |     if ( (twst != TW_MT_SLA_ACK) && (twst != TW_MR_SLA_ACK) ) return 1; | ||||||
|  | 
 | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint8_t i2c_write(uint8_t data) | ||||||
|  | { | ||||||
|  |     // load data into data register
 | ||||||
|  |     TWDR = data; | ||||||
|  |     // start transmission of data
 | ||||||
|  |     TWCR = (1 << TWINT) | (1 << TWEN); | ||||||
|  |     // wait for end of transmission
 | ||||||
|  |     while (!(TWCR & (1 << TWINT))) | ||||||
|  |         ; | ||||||
|  | 
 | ||||||
|  |     if ((TWSR & 0xF8) != TW_MT_DATA_ACK) { | ||||||
|  |         return 1; | ||||||
|  |     } | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint8_t i2c_transmit(uint8_t address, uint8_t* data, uint16_t length) | ||||||
|  | { | ||||||
|  |     TWBR = (uint8_t)TWBR_val; | ||||||
|  |     if (i2c_start(address | I2C_WRITE)) | ||||||
|  |         return 1; | ||||||
|  |     for (uint16_t i = 0; i < length; i++) { | ||||||
|  |         if (i2c_write(data[i])) | ||||||
|  |             return 1; | ||||||
|  |     } | ||||||
|  |     // transmit STOP condition
 | ||||||
|  |     TWCR = (1 << TWINT) | (1 << TWEN) | (1 << TWSTO); | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void setFrame(uint8_t device, uint8_t frame) | ||||||
|  | { | ||||||
|  |     static uint8_t current_frame = -1; | ||||||
|  |     if(current_frame != frame){ | ||||||
|  |         uint8_t payload[] = { | ||||||
|  |             ISSI_ADDR_DEFAULT | device << 1, | ||||||
|  |             ISSI_COMMANDREGISTER, | ||||||
|  |             frame | ||||||
|  |         }; | ||||||
|  |         TWITransmitData(payload, sizeof(payload), 0, 1); | ||||||
|  |     } | ||||||
|  |     // static uint8_t current_frame = 0xFF;
 | ||||||
|  |     // if(current_frame == frame){
 | ||||||
|  |     //     // return;
 | ||||||
|  |     // }
 | ||||||
|  |     // uint8_t payload[2] = { ISSI_COMMANDREGISTER, frame };
 | ||||||
|  |     // i2c_transmit(ISSI_ADDR_DEFAULT | device << 1, payload, 2);
 | ||||||
|  |     // current_frame = frame;
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void writeRegister8(uint8_t device, uint8_t frame, uint8_t reg, uint8_t data) | ||||||
|  | { | ||||||
|  |     // Set the frame
 | ||||||
|  |     setFrame(device, frame); | ||||||
|  | 
 | ||||||
|  |     // Write to the register
 | ||||||
|  |     uint8_t payload[] = { | ||||||
|  |         ISSI_ADDR_DEFAULT | device << 1, | ||||||
|  |         reg, | ||||||
|  |         data | ||||||
|  |     }; | ||||||
|  |     TWITransmitData(payload, sizeof(payload), 0, 1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // void activateLED(uint8_t matrix, uint8_t cx, uint8_t cy, uint8_t pwm)
 | ||||||
|  | // {
 | ||||||
|  | //     xprintf("activeLED: %02X %02X %02X %02X\n", matrix, cy, cx, pwm);
 | ||||||
|  | //     uint8_t x = cx - 1;  // funciton takes 1 based counts, but we need 0...
 | ||||||
|  | //     uint8_t y = cy - 1;  // creating them once for less confusion
 | ||||||
|  | //     if(pwm == 0){
 | ||||||
|  | //         cbi(control[matrix][y], x);
 | ||||||
|  | //     }else{
 | ||||||
|  | //         sbi(control[matrix][y], x);
 | ||||||
|  | //     }
 | ||||||
|  | //     uint8_t device = (matrix & 0x06) >> 1;
 | ||||||
|  | //     uint8_t control_reg = (y << 1) | (matrix & 0x01);
 | ||||||
|  | //     uint8_t pwm_reg = 0;
 | ||||||
|  | //     switch(matrix & 0x01){
 | ||||||
|  | //         case 0:
 | ||||||
|  | //             pwm_reg = 0x24;
 | ||||||
|  | //             break;
 | ||||||
|  | //         case 1:
 | ||||||
|  | //             pwm_reg = 0x2C;
 | ||||||
|  | //             break;
 | ||||||
|  | //     }
 | ||||||
|  | //     pwm_reg += (y << 4) + x;
 | ||||||
|  | //     xprintf("  device: %02X\n", device);
 | ||||||
|  | //     xprintf("  control: %02X %02X\n", control_reg, control[matrix][y]);
 | ||||||
|  | //     xprintf("  pwm:     %02X %02X\n", pwm_reg, pwm);
 | ||||||
|  | //     writeRegister8(device, 0, control_reg, control[matrix][y]);
 | ||||||
|  | //     writeRegister8(device, 0, control_reg + 0x12, control[matrix][y]);
 | ||||||
|  | //     writeRegister8(device, 0, pwm_reg, pwm);
 | ||||||
|  | // }
 | ||||||
|  | 
 | ||||||
|  | void activateLED(uint8_t matrix, uint8_t cx, uint8_t cy, uint8_t pwm) | ||||||
|  | { | ||||||
|  |     uint8_t device_addr = (matrix & 0x06) >> 1; | ||||||
|  |     ISSIDeviceStruct *device = issi_devices[device_addr]; | ||||||
|  |     if(device == 0){ | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     // xprintf("activeLED: %02X %02X %02X %02X\n", matrix, cy, cx, pwm);
 | ||||||
|  |     uint8_t x = cx - 1;  // funciton takes 1 based counts, but we need 0...
 | ||||||
|  |     uint8_t y = cy - 1;  // creating them once for less confusion
 | ||||||
|  |     uint8_t control_reg = (y << 1) | (matrix & 0x01); | ||||||
|  |     if(pwm == 0){ | ||||||
|  |         cbi(device->led_ctrl[control_reg], x); | ||||||
|  |         cbi(device->led_blink_ctrl[control_reg], x); | ||||||
|  |      }else{ | ||||||
|  |         sbi(device->led_ctrl[control_reg], x); | ||||||
|  |         sbi(device->led_blink_ctrl[control_reg], x); | ||||||
|  |     } | ||||||
|  |     uint8_t pwm_reg = 0; | ||||||
|  |     switch(matrix & 0x01){ | ||||||
|  |         case 0: | ||||||
|  |             pwm_reg = 0x00; | ||||||
|  |             break; | ||||||
|  |         case 1: | ||||||
|  |             pwm_reg = 0x08; | ||||||
|  |             break; | ||||||
|  |     } | ||||||
|  |     pwm_reg += (y << 4) + x; | ||||||
|  |     // xprintf("  device_addr: %02X\n", device_addr);
 | ||||||
|  |     // xprintf("  control: %02X %02X\n", control_reg, control[matrix][y]);
 | ||||||
|  |     // xprintf("  pwm:     %02X %02X\n", pwm_reg, pwm);
 | ||||||
|  |     // writeRegister8(device_addr, 0, control_reg, control[matrix][y]);
 | ||||||
|  |     device->led_pwm[pwm_reg] = pwm; | ||||||
|  |     device->led_dirty = 1; | ||||||
|  | 
 | ||||||
|  |     // writeRegister8(device_addr, 0, control_reg + 0x12, control[matrix][y]);
 | ||||||
|  |     // writeRegister8(device_addr, 0, pwm_reg, pwm);
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void update_issi(uint8_t device_addr, uint8_t blocking) | ||||||
|  | { | ||||||
|  |     // This seems to take about 6ms
 | ||||||
|  |     ISSIDeviceStruct *device = issi_devices[device_addr]; | ||||||
|  |     if(device != 0){ | ||||||
|  |         if(device->fn_dirty){ | ||||||
|  |             device->fn_dirty = 0; | ||||||
|  |             setFrame(device_addr, ISSI_BANK_FUNCTIONREG); | ||||||
|  |             TWITransmitData(&device->fn_device_addr, sizeof(device->fn_registers) + 2, 0, 1); | ||||||
|  |         } | ||||||
|  |         if(device->led_dirty){ | ||||||
|  |             device->led_dirty = 0; | ||||||
|  |             setFrame(device_addr, 0); | ||||||
|  |             TWITransmitData(&device->led_device_addr, 0xB6, 0, blocking); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void issi_init(void) | ||||||
|  | { | ||||||
|  |     // Set LED_EN/SDB high to enable the chip
 | ||||||
|  |     xprintf("Enabing SDB on pin: %d\n", LED_EN_PIN); | ||||||
|  |     _SFR_IO8((LED_EN_PIN >> 4) + 1) &= ~_BV(LED_EN_PIN & 0xF); // IN
 | ||||||
|  |     _SFR_IO8((LED_EN_PIN >> 4) + 2) |=  _BV(LED_EN_PIN & 0xF); // HI
 | ||||||
|  |     TWIInit(); | ||||||
|  |     for(uint8_t device_addr = 0; device_addr < 4; device_addr++){ | ||||||
|  |         xprintf("ISSI Init device: %d\n", device_addr); | ||||||
|  |         // If this device has been previously allocated, free it
 | ||||||
|  |         if(issi_devices[device_addr] != 0){ | ||||||
|  |             free(issi_devices[device_addr]); | ||||||
|  |         } | ||||||
|  |         // Try to shutdown the device, if this fails skip this device
 | ||||||
|  |         writeRegister8(device_addr, ISSI_BANK_FUNCTIONREG, ISSI_REG_SHUTDOWN, 0x00); | ||||||
|  |         while (!isTWIReady()){_delay_us(1);} | ||||||
|  |         if(TWIInfo.errorCode != 0xFF){ | ||||||
|  |             xprintf("ISSI init failed %d %02X %02X\n", device_addr, TWIInfo.mode, TWIInfo.errorCode); | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |         // Allocate the device structure - calloc zeros it for us
 | ||||||
|  |         ISSIDeviceStruct *device = (ISSIDeviceStruct *)calloc(sizeof(ISSIDeviceStruct) * 2, 1); | ||||||
|  |         issi_devices[device_addr] = device; | ||||||
|  |         device->fn_device_addr = ISSI_ADDR_DEFAULT | device_addr << 1; | ||||||
|  |         device->fn_register_addr = 0; | ||||||
|  |         device->led_device_addr = ISSI_ADDR_DEFAULT | device_addr << 1; | ||||||
|  |         device->led_register_addr = 0; | ||||||
|  |         // set dirty bits so that all of the buffered data is written out
 | ||||||
|  |         device->fn_dirty = 1; | ||||||
|  |         device->led_dirty = 1; | ||||||
|  |         update_issi(device_addr, 1); | ||||||
|  |         // Set the function register to picture mode
 | ||||||
|  |         // device->fn_reg[ISSI_REG_CONFIG] = ISSI_REG_CONFIG_PICTUREMODE;
 | ||||||
|  |         writeRegister8(device_addr, ISSI_BANK_FUNCTIONREG, ISSI_REG_SHUTDOWN, 0x01); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Shutdown and set all registers to 0
 | ||||||
|  |     // writeRegister8(device_addr, ISSI_BANK_FUNCTIONREG, ISSI_REG_SHUTDOWN, 0x00);
 | ||||||
|  |     // for(uint8_t bank = 0; bank <= 7; bank++){
 | ||||||
|  |     //     for (uint8_t reg = 0x00; reg <= 0xB3; reg++) {
 | ||||||
|  |     //         writeRegister8(device_addr, bank, reg, 0x00);
 | ||||||
|  |     //     }
 | ||||||
|  |     // }
 | ||||||
|  |     // for (uint8_t reg = 0; reg <= 0x0C; reg++) {
 | ||||||
|  |     //     writeRegister8(device_addr, ISSI_BANK_FUNCTIONREG, reg, 0x00);
 | ||||||
|  |     // }
 | ||||||
|  |     // writeRegister8(device_addr, ISSI_BANK_FUNCTIONREG, ISSI_REG_CONFIG, ISSI_REG_CONFIG_PICTUREMODE);
 | ||||||
|  |     // writeRegister8(device_addr, ISSI_BANK_FUNCTIONREG, ISSI_REG_SHUTDOWN, 0x01);
 | ||||||
|  |     // picture mode
 | ||||||
|  |     // writeRegister8(ISSI_BANK_FUNCTIONREG, 0x01, 0x01);
 | ||||||
|  | 
 | ||||||
|  |     //Enable blink
 | ||||||
|  |     // writeRegister8(ISSI_BANK_FUNCTIONREG, 0x05, 0x48B);
 | ||||||
|  | 
 | ||||||
|  |     //Enable Breath
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										40
									
								
								keyboards/meira/issi.h
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										40
									
								
								keyboards/meira/issi.h
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,40 @@ | ||||||
|  | #ifdef ISSI_ENABLE | ||||||
|  | #ifndef ISSI_H | ||||||
|  | #define ISSI_H | ||||||
|  | 
 | ||||||
|  | typedef struct ISSIDeviceStruct{ | ||||||
|  |     uint8_t fn_dirty;       // function registers need to be resent
 | ||||||
|  |     uint8_t fn_device_addr; | ||||||
|  |     uint8_t fn_register_addr; | ||||||
|  |     uint8_t fn_registers[13]; | ||||||
|  |     uint8_t led_dirty;      // LED data has changed and needs to be resent
 | ||||||
|  |     uint8_t led_device_addr; | ||||||
|  |     uint8_t led_register_addr; | ||||||
|  |     uint8_t led_ctrl[18]; | ||||||
|  |     uint8_t led_blink_ctrl[18]; | ||||||
|  |     uint8_t led_pwm[144]; | ||||||
|  | }ISSIDeviceStruct; | ||||||
|  | 
 | ||||||
|  | extern ISSIDeviceStruct *issi_devices[]; | ||||||
|  | 
 | ||||||
|  | // Low level commands- 'device' is the 2-bit i2c id.
 | ||||||
|  | void issi_init(void); | ||||||
|  | void set_shutdown(uint8_t device, uint8_t shutdown); | ||||||
|  | void writeRegister8(uint8_t device, uint8_t frame, uint8_t reg, uint8_t data); | ||||||
|  | 
 | ||||||
|  | // Higher level, no device is given, but it is calculated from 'matrix'
 | ||||||
|  | // Each device has 2 blocks, max of 4 devices:
 | ||||||
|  | //  Device  |   Block   =   Matrix
 | ||||||
|  | //    0           A           0
 | ||||||
|  | //    0           B           1
 | ||||||
|  | //    1           A           2
 | ||||||
|  | //    1           B           3
 | ||||||
|  | //    2           A           4
 | ||||||
|  | //    2           B           5
 | ||||||
|  | //    3           A           6
 | ||||||
|  | //    3           B           7
 | ||||||
|  | void activateLED(uint8_t matrix, uint8_t cx, uint8_t cy, uint8_t pwm); | ||||||
|  | void update_issi(uint8_t device_addr, uint8_t blocking); | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
|  | #endif | ||||||
							
								
								
									
										24
									
								
								keyboards/meira/keymaps/default/config.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								keyboards/meira/keymaps/default/config.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,24 @@ | ||||||
|  | /* Copyright 2017 REPLACE_WITH_YOUR_NAME
 | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef CONFIG_USER_H | ||||||
|  | #define CONFIG_USER_H | ||||||
|  | 
 | ||||||
|  | #include "../../config.h" | ||||||
|  | 
 | ||||||
|  | // place overrides here
 | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										320
									
								
								keyboards/meira/keymaps/default/keymap.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										320
									
								
								keyboards/meira/keymaps/default/keymap.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,320 @@ | ||||||
|  | /* Copyright 2017 Cole Markham
 | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  |  */ | ||||||
|  | #include "meira.h" | ||||||
|  | #include "issi.h" | ||||||
|  | #include "lighting.h" | ||||||
|  | 
 | ||||||
|  | #ifdef RGBLIGHT_ENABLE | ||||||
|  | //Following line allows macro to read current RGB settings
 | ||||||
|  | extern rgblight_config_t rgblight_config; | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #define _QWERTY 0 | ||||||
|  | #define _COLEMAK 1 | ||||||
|  | #define _DVORAK 2 | ||||||
|  | #define _LOWER 3 | ||||||
|  | #define _RAISE 4 | ||||||
|  | #define _ADJUST 16 | ||||||
|  | 
 | ||||||
|  | enum custom_keycodes { | ||||||
|  |   QWERTY = SAFE_RANGE, | ||||||
|  |   COLEMAK, | ||||||
|  |   DVORAK, | ||||||
|  |   LOWER, | ||||||
|  |   RAISE, | ||||||
|  |   ADJUST, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // define variables for reactive RGB
 | ||||||
|  | bool TOG_STATUS = false; | ||||||
|  | int RGB_current_mode; | ||||||
|  | 
 | ||||||
|  | const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = { | ||||||
|  |         /* Qwerty
 | ||||||
|  |          * ,-----------------------------------------------------------------------------------. | ||||||
|  |          * | Esc  |   Q  |   W  |   E  |   R  |   T  |   Y  |   U  |   I  |   O  |   P  | Bksp | | ||||||
|  |          * |------+------+------+------+------+-------------+------+------+------+------+------| | ||||||
|  |          * | Tab  |   A  |   S  |   D  |   F  |   G  |   H  |   J  |   K  |   L  |   ;  |  '   | | ||||||
|  |          * |------+------+------+------+------+------|------+------+------+------+------+------| | ||||||
|  |          * | Shift|   Z  |   X  |   C  |   V  |   B  |   N  |   M  |   ,  |   .  |   /  |Enter | | ||||||
|  |          * |------+------+------+------+------+------+------+------+------+------+------+------| | ||||||
|  |          * |Adjust| Ctrl | Ctrl  | Alt  |Lower | Cmd  |Space |Raise | Left | Down |  Up  |Right | | ||||||
|  |          * `-----------------------------------------------------------------------------------' | ||||||
|  |          */ | ||||||
|  |         [_QWERTY] = KEYMAP( \ | ||||||
|  |                 KC_ESC,  KC_Q,    KC_W,    KC_E,    KC_R,    KC_T,    KC_Y,    KC_U,    KC_I,    KC_O,    KC_P,    KC_BSPC, \ | ||||||
|  |                 KC_TAB,  KC_A,    KC_S,    KC_D,    KC_F,    KC_G,    KC_H,    KC_J,    KC_K,    KC_L,    KC_SCLN, KC_QUOT, \ | ||||||
|  |                 KC_LSFT, KC_Z,    KC_X,    KC_C,    KC_V,    KC_B,    KC_N,    KC_M,    KC_COMM, KC_DOT,  KC_SLSH,  KC_ENT, \ | ||||||
|  |                 ADJUST,  KC_LCTL, KC_LALT, KC_LALT, LOWER,   KC_LGUI,  KC_SPC,  RAISE,   KC_LEFT, KC_DOWN, KC_UP,   KC_RGHT \ | ||||||
|  |         ), | ||||||
|  | 
 | ||||||
|  |         /* Colemak
 | ||||||
|  |          * ,-----------------------------------------------------------------------------------. | ||||||
|  |          * | Tab  |   Q  |   W  |   F  |   P  |   G  |   J  |   L  |   U  |   Y  |   ;  | Bksp | | ||||||
|  |          * |------+------+------+------+------+-------------+------+------+------+------+------| | ||||||
|  |          * | Esc  |   A  |   R  |   S  |   T  |   D  |   H  |   N  |   E  |   I  |   O  |  "   | | ||||||
|  |          * |------+------+------+------+------+------|------+------+------+------+------+------| | ||||||
|  |          * | Shift|   Z  |   X  |   C  |   V  |   B  |   K  |   M  |   ,  |   .  |   /  |Enter | | ||||||
|  |          * |------+------+------+------+------+------+------+------+------+------+------+------| | ||||||
|  |          * |Adjust| Ctrl | Alt  | GUI  |Lower |Space |Space |Raise | Left | Down |  Up  |Right | | ||||||
|  |          * `-----------------------------------------------------------------------------------' | ||||||
|  |          */ | ||||||
|  |         [_COLEMAK] = KEYMAP( \ | ||||||
|  |                 KC_TAB,  KC_Q,    KC_W,    KC_F,    KC_P,    KC_G,    KC_J,    KC_L,    KC_U,    KC_Y,    KC_SCLN, KC_BSPC, \ | ||||||
|  |                 KC_ESC,  KC_A,    KC_R,    KC_S,    KC_T,    KC_D,    KC_H,    KC_N,    KC_E,    KC_I,    KC_O,    KC_QUOT, \ | ||||||
|  |                 KC_LSFT, KC_Z,    KC_X,    KC_C,    KC_V,    KC_B,    KC_K,    KC_M,    KC_COMM, KC_DOT,  KC_SLSH, KC_ENT , \ | ||||||
|  |                 ADJUST,  KC_LCTL, KC_LALT, KC_LGUI, LOWER,   KC_SPC,  KC_SPC,  RAISE,   KC_LEFT, KC_DOWN, KC_UP,   KC_RGHT \ | ||||||
|  |         ), | ||||||
|  | 
 | ||||||
|  |         /* Dvorak
 | ||||||
|  |          * ,-----------------------------------------------------------------------------------. | ||||||
|  |          * | Tab  |   "  |   ,  |   .  |   P  |   Y  |   F  |   G  |   C  |   R  |   L  | Bksp | | ||||||
|  |          * |------+------+------+------+------+-------------+------+------+------+------+------| | ||||||
|  |          * | Esc  |   A  |   O  |   E  |   U  |   I  |   D  |   H  |   T  |   N  |   S  |  /   | | ||||||
|  |          * |------+------+------+------+------+------|------+------+------+------+------+------| | ||||||
|  |          * | Shift|   ;  |   Q  |   J  |   K  |   X  |   B  |   M  |   W  |   V  |   Z  |Enter | | ||||||
|  |          * |------+------+------+------+------+------+------+------+------+------+------+------| | ||||||
|  |          * |Adjust| Ctrl | Alt  | GUI  |Lower |Space |Space |Raise | Left | Down |  Up  |Right | | ||||||
|  |          * `-----------------------------------------------------------------------------------' | ||||||
|  |          */ | ||||||
|  |         [_DVORAK] = KEYMAP( \ | ||||||
|  |                 KC_TAB,  KC_QUOT, KC_COMM, KC_DOT,  KC_P,    KC_Y,    KC_F,    KC_G,    KC_C,    KC_R,    KC_L,    KC_BSPC, \ | ||||||
|  |                 KC_ESC,  KC_A,    KC_O,    KC_E,    KC_U,    KC_I,    KC_D,    KC_H,    KC_T,    KC_N,    KC_S,    KC_SLSH, \ | ||||||
|  |                 KC_LSFT, KC_SCLN, KC_Q,    KC_J,    KC_K,    KC_X,    KC_B,    KC_M,    KC_W,    KC_V,    KC_Z,    KC_ENT , \ | ||||||
|  |                 ADJUST,  KC_LCTL, KC_LALT, KC_LGUI, LOWER,   KC_SPC,  KC_SPC,  RAISE,   KC_LEFT, KC_DOWN, KC_UP,   KC_RGHT \ | ||||||
|  |         ), | ||||||
|  | 
 | ||||||
|  |         /* Lower
 | ||||||
|  |          * ,-----------------------------------------------------------------------------------. | ||||||
|  |          * |      |   !  |   @  |   #  |   $  |   %  |   ^  |   &  |   *  |   (  |   )  | Bksp | | ||||||
|  |          * |------+------+------+------+------+-------------+------+------+------+------+------| | ||||||
|  |          * |   ~  |  F1  |  F2  |  F3  |  F4  |  F5  |  F6  |   _  |   +  |     |    \  |  |   | | ||||||
|  |          * |------+------+------+------+------+------|------+------+------+------+------+------| | ||||||
|  |          * |      |  F7  |  F8  |  F9  |  F10 |  F11 |  F12 |ISO ~ |ISO | |      |      |Enter | | ||||||
|  |          * |------+------+------+------+------+------+------+------+------+------+------+------| | ||||||
|  |          * |      |      |      |      |      |             |      | Next | Vol- | Vol+ | Play | | ||||||
|  |          * `-----------------------------------------------------------------------------------' | ||||||
|  |          */ | ||||||
|  |         [_LOWER] = KEYMAP( \ | ||||||
|  |                 _______, KC_EXLM, KC_AT,   KC_HASH, KC_DLR,  KC_PERC, KC_CIRC, KC_AMPR, KC_ASTR, KC_LPRN, KC_RPRN, KC_DEL, \ | ||||||
|  |                 KC_TILD,  KC_F1,   KC_F2,   KC_F3,   KC_F4,   KC_F5,   KC_F6,   KC_UNDS, KC_PLUS, KC_LCBR, KC_RCBR, KC_PIPE, \ | ||||||
|  |                 _______, KC_F7,   KC_F8,   KC_F9,   KC_F10,  KC_F11,  KC_F12,S(KC_NUHS),S(KC_NUBS),_______, _______, KC_QUOT, \ | ||||||
|  |                 _______, _______, _______, _______, _______, _______, _______, _______, KC_HOME, KC_PGDN, KC_PGUP, KC_END \ | ||||||
|  |         ), | ||||||
|  | 
 | ||||||
|  |         /* Raise
 | ||||||
|  |          * ,-----------------------------------------------------------------------------------. | ||||||
|  |          * |      |   1  |   2  |   3  |   4  |   5  |   6  |   7  |   8  |   9  |   0  | Bksp | | ||||||
|  |          * |------+------+------+------+------+-------------+------+------+------+------+------| | ||||||
|  |          * |   `  |  F1  |  F2  |  F3  |  F4  |  F5  |  F6  |   -  |   =  |   [  |   ]  |  \   | | ||||||
|  |          * |------+------+------+------+------+------|------+------+------+------+------+------| | ||||||
|  |          * |      |  F7  |  F8  |  F9  |  F10 |  F11 |  F12 |ISO # |ISO / |      |      |Enter | | ||||||
|  |          * |------+------+------+------+------+------+------+------+------+------+------+------| | ||||||
|  |          * |      |      |      |      |      |             |      | Home | PgUp | PgDn | End  | | ||||||
|  |          * `-----------------------------------------------------------------------------------' | ||||||
|  |          */ | ||||||
|  |         [_RAISE] = KEYMAP( \ | ||||||
|  |                 _______,  KC_1,    KC_2,    KC_3,    KC_4,    KC_5,    KC_6,    KC_7,    KC_8,    KC_9,    KC_0,    KC_DEL, \ | ||||||
|  |                 KC_GRV,  KC_F1,   KC_F2,   KC_F3,   KC_F4,   KC_F5,   KC_F6,   KC_MINS, KC_EQL,  KC_LBRC, KC_RBRC, KC_BSLS, \ | ||||||
|  |                 _______, KC_F7,   KC_F8,   KC_F9,   KC_F10,  KC_F11,  KC_F12,  KC_NUHS, KC_NUBS, _______, _______, _______, \ | ||||||
|  |                 _______, _______, _______, _______, _______, _______, _______, _______, KC_HOME, KC_PGDN, KC_PGUP, KC_END \ | ||||||
|  |         ), | ||||||
|  | 
 | ||||||
|  |         /* Adjust (Lower + Raise)
 | ||||||
|  |          * ,-----------------------------------------------------------------------------------. | ||||||
|  |          * |      | Reset|      |      |      |      |      |      |      |      |      |  Del | | ||||||
|  |          * |------+------+------+------+------+-------------+------+------+------+------+------| | ||||||
|  |          * |      |      |      |Aud on|Audoff|AGnorm|AGswap|Qwerty|Colemk|Dvorak|      |      | | ||||||
|  |          * |------+------+------+------+------+------|------+------+------+------+------+------| | ||||||
|  |          * |      |      |      |      |      |      |      |      |      |      |      |      | | ||||||
|  |          * |------+------+------+------+------+------+------+------+------+------+------+------| | ||||||
|  |          * |      |      |      |      |      |             |      |      |      |      |      | | ||||||
|  |          * `-----------------------------------------------------------------------------------' | ||||||
|  |          */ | ||||||
|  |         [_ADJUST] =  KEYMAP( \ | ||||||
|  |                 BL_TOGG, RESET,   _______, KC_MRWD, KC_MPLY, KC_MFFD, KC_PSCR, _______, KC_MUTE, KC_VOLD, KC_VOLU, KC_DEL, \ | ||||||
|  |                 BL_STEP, RGB_MOD, _______, AU_ON,   AU_OFF,  AG_NORM, AG_SWAP, QWERTY,  COLEMAK, DVORAK,  _______, _______, \ | ||||||
|  |                 _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, \ | ||||||
|  |                 _______, KC_PSCR, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______ \ | ||||||
|  |         ) | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const uint16_t PROGMEM fn_actions[] = { | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Setting ADJUST layer RGB back to default
 | ||||||
|  | void update_tri_layer_RGB(uint8_t layer1, uint8_t layer2, uint8_t layer3) { | ||||||
|  |   if (IS_LAYER_ON(layer1) && IS_LAYER_ON(layer2)) { | ||||||
|  | #ifdef RGBLIGHT_ENABLE | ||||||
|  |     rgblight_mode(RGB_current_mode); | ||||||
|  | #endif | ||||||
|  |     layer_on(layer3); | ||||||
|  |   } else { | ||||||
|  |     layer_off(layer3); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const macro_t *action_get_macro(keyrecord_t *record, uint8_t id, uint8_t opt) | ||||||
|  | { | ||||||
|  |   // MACRODOWN only works in this function
 | ||||||
|  |       switch(id) { | ||||||
|  |         case 0: | ||||||
|  |           if (record->event.pressed) { | ||||||
|  |             register_code(KC_RSFT); | ||||||
|  |           } else { | ||||||
|  |             unregister_code(KC_RSFT); | ||||||
|  |           } | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |     return MACRO_NONE; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void matrix_init_user(void) { | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void matrix_scan_user(void) { | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool process_record_user(uint16_t keycode, keyrecord_t *record) { | ||||||
|  |   switch (keycode) { | ||||||
|  |     case QWERTY: | ||||||
|  |       if (record->event.pressed) { | ||||||
|  |         #ifdef AUDIO_ENABLE | ||||||
|  |           PLAY_NOTE_ARRAY(tone_qwerty, false, 0); | ||||||
|  |         #endif | ||||||
|  | //        persistent_default_layer_set(1UL<<_QWERTY);
 | ||||||
|  |       } | ||||||
|  |       return false; | ||||||
|  |       break; | ||||||
|  |     case COLEMAK: | ||||||
|  |       if (record->event.pressed) { | ||||||
|  |         #ifdef AUDIO_ENABLE | ||||||
|  |           PLAY_NOTE_ARRAY(tone_colemak, false, 0); | ||||||
|  |         #endif | ||||||
|  | //        persistent_default_layer_set(1UL<<_COLEMAK);
 | ||||||
|  |       } | ||||||
|  |       return false; | ||||||
|  |       break; | ||||||
|  |     case DVORAK: | ||||||
|  |       if (record->event.pressed) { | ||||||
|  |         #ifdef AUDIO_ENABLE | ||||||
|  |           PLAY_NOTE_ARRAY(tone_dvorak, false, 0); | ||||||
|  |         #endif | ||||||
|  | //        persistent_default_layer_set(1UL<<_DVORAK);
 | ||||||
|  |       } | ||||||
|  |       return false; | ||||||
|  |       break; | ||||||
|  |     case LOWER: | ||||||
|  |         if (record->event.pressed) { | ||||||
|  |             //not sure how to have keyboard check mode and set it to a variable, so my work around
 | ||||||
|  |             //uses another variable that would be set to true after the first time a reactive key is pressed.
 | ||||||
|  |             if (TOG_STATUS) { //TOG_STATUS checks is another reactive key currently pressed, only changes RGB mode if returns false
 | ||||||
|  |             } else { | ||||||
|  |                 TOG_STATUS = !TOG_STATUS; | ||||||
|  | #ifdef RGBLIGHT_ENABLE | ||||||
|  |                 rgblight_mode(16); | ||||||
|  | #endif | ||||||
|  |             } | ||||||
|  |             layer_on(_LOWER); | ||||||
|  |             update_tri_layer_RGB(_LOWER, _RAISE, _ADJUST); | ||||||
|  |         } else { | ||||||
|  | #ifdef RGBLIGHT_ENABLE | ||||||
|  |             rgblight_mode(RGB_current_mode);   // revert RGB to initial mode prior to RGB mode change
 | ||||||
|  | #endif | ||||||
|  |             TOG_STATUS = false; | ||||||
|  |             layer_off(_LOWER); | ||||||
|  |             update_tri_layer_RGB(_LOWER, _RAISE, _ADJUST); | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |         break; | ||||||
|  |     case RAISE: | ||||||
|  |         if (record->event.pressed) { | ||||||
|  |             //not sure how to have keyboard check mode and set it to a variable, so my work around
 | ||||||
|  |             //uses another variable that would be set to true after the first time a reactive key is pressed.
 | ||||||
|  |             if (TOG_STATUS) { //TOG_STATUS checks is another reactive key currently pressed, only changes RGB mode if returns false
 | ||||||
|  |             } else { | ||||||
|  |                 TOG_STATUS = !TOG_STATUS; | ||||||
|  | #ifdef RGBLIGHT_ENABLE | ||||||
|  |                 rgblight_mode(15); | ||||||
|  | #endif | ||||||
|  |             } | ||||||
|  |             layer_on(_RAISE); | ||||||
|  |             update_tri_layer_RGB(_LOWER, _RAISE, _ADJUST); | ||||||
|  |         } else { | ||||||
|  | #ifdef RGBLIGHT_ENABLE | ||||||
|  |             rgblight_mode(RGB_current_mode);  // revert RGB to initial mode prior to RGB mode change
 | ||||||
|  | #endif | ||||||
|  |             layer_off(_RAISE); | ||||||
|  |             TOG_STATUS = false; | ||||||
|  |             update_tri_layer_RGB(_LOWER, _RAISE, _ADJUST); | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |         break; | ||||||
|  |     case ADJUST: | ||||||
|  |         // FIXME add RGB feedback
 | ||||||
|  |         if (record->event.pressed) { | ||||||
|  |             layer_on(_ADJUST); | ||||||
|  |         } else { | ||||||
|  |             layer_off(_ADJUST); | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |         break; | ||||||
|  |     case BL_TOGG: | ||||||
|  | #ifdef ISSI_ENABLE | ||||||
|  |         if (record->event.pressed) { | ||||||
|  |             print("Enabling backlight\n"); | ||||||
|  |             issi_init(); | ||||||
|  |         } | ||||||
|  | #endif | ||||||
|  |         return false; | ||||||
|  |         break; | ||||||
|  |     case BL_STEP: | ||||||
|  |         if (record->event.pressed) { | ||||||
|  |             print("Stepping backlight\n"); | ||||||
|  | #ifdef BACKLIGHT_ENABLE | ||||||
|  |             print("Really stepping backlight\n"); | ||||||
|  |             backlight_step(); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |         break; | ||||||
|  |         //led operations - RGB mode change now updates the RGB_current_mode to allow the right RGB mode to be set after reactive keys are released
 | ||||||
|  | #ifdef RGBLIGHT_ENABLE | ||||||
|  |     case RGB_MOD: | ||||||
|  |         if (record->event.pressed) { | ||||||
|  |             rgblight_mode(RGB_current_mode); | ||||||
|  |             rgblight_step(); | ||||||
|  |             RGB_current_mode = rgblight_config.mode; | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |         break; | ||||||
|  | #endif | ||||||
|  | //    case BL_INC:
 | ||||||
|  | //    		meira_inc_backlight_level();
 | ||||||
|  | //    	      return false;
 | ||||||
|  | //    		break;
 | ||||||
|  |   } | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void led_set_user(uint8_t usb_led) { | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								keyboards/meira/keymaps/default/readme.md
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								keyboards/meira/keymaps/default/readme.md
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | ||||||
|  | # The default keymap for meira | ||||||
							
								
								
									
										95
									
								
								keyboards/meira/lighting.c
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										95
									
								
								keyboards/meira/lighting.c
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,95 @@ | ||||||
|  | #ifdef ISSI_ENABLE | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #include <avr/sfr_defs.h> | ||||||
|  | #include <avr/timer_avr.h> | ||||||
|  | #include <avr/wdt.h> | ||||||
|  | #include "meira.h" | ||||||
|  | #include "issi.h" | ||||||
|  | #include "TWIlib.h" | ||||||
|  | #include "lighting.h" | ||||||
|  | #include "debug.h" | ||||||
|  | #include "audio/audio.h" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | const uint8_t backlight_pwm_map[BACKLIGHT_LEVELS] = BACKLIGHT_PWM_MAP; | ||||||
|  | 
 | ||||||
|  |     const uint8_t switch_matrices[] = {0, 1}; | ||||||
|  | 
 | ||||||
|  |     // Maps switch LEDs from Row/Col to ISSI matrix.
 | ||||||
|  |     // Value breakdown:
 | ||||||
|  |     //     Bit     | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
 | ||||||
|  |     //             |   | ISSI Col  |    ISSI Row   |
 | ||||||
|  |     //             /   |
 | ||||||
|  |     //             Device
 | ||||||
|  | //    const uint8_t switch_leds[MATRIX_ROWS][MATRIX_COLS] =
 | ||||||
|  | //    KEYMAP(
 | ||||||
|  | //      0x29, 0x28, 0x27, 0x26, 0x25, 0x24, 0x23, 0x22, 0x21, 0xA9, 0xA8, 0xA7, 0xA6, 0xA5,
 | ||||||
|  | //      0x39, 0x38, 0x37, 0x36, 0x35, 0x34, 0x33, 0x32, 0x31, 0xB9, 0xB8, 0xB7, 0xB6, 0xB5,
 | ||||||
|  | //      0x49, 0x48, 0x47, 0x45, 0x44, 0x43, 0x42, 0x41, 0xC9, 0xC8, 0xC7, 0xC6,
 | ||||||
|  | //      0x59, 0x58, 0x57, 0x56,       0x55,       0x51, 0xD6, 0xE5, 0xE4, 0xE3, 0xE2);
 | ||||||
|  | 
 | ||||||
|  | void backlight_set(uint8_t level){ | ||||||
|  | #ifdef BACKLIGHT_ENABLE | ||||||
|  |     uint8_t pwm_value = 0; | ||||||
|  |     if(level >= BACKLIGHT_LEVELS){ | ||||||
|  |         level = BACKLIGHT_LEVELS; | ||||||
|  |     } | ||||||
|  |     if(level > 0){ | ||||||
|  |         pwm_value = backlight_pwm_map[level-1]; | ||||||
|  |     } | ||||||
|  |     xprintf("BACKLIGHT_LEVELS: %d\n", BACKLIGHT_LEVELS); | ||||||
|  |     xprintf("backlight_set level: %d pwm: %d\n", level, pwm_value); | ||||||
|  |     for(int x = 1; x <= 9; x++){ | ||||||
|  |         for(int y = 1; y <= 9; y++){ | ||||||
|  |             activateLED(switch_matrices[0], x, y, pwm_value); | ||||||
|  |             activateLED(switch_matrices[1], x, y, pwm_value); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void set_backlight_by_keymap(uint8_t col, uint8_t row){ | ||||||
|  | //    dprintf("LED: %02X, %d %d %d\n", lookup_value, matrix, led_col, led_row);
 | ||||||
|  | //    activateLED(matrix, led_col, led_row, 255);
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void force_issi_refresh(){ | ||||||
|  |     issi_devices[0]->led_dirty = true; | ||||||
|  |     update_issi(0, true); | ||||||
|  |     issi_devices[3]->led_dirty = true; | ||||||
|  |     update_issi(3, true); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void led_test(){ | ||||||
|  | #ifdef WATCHDOG_ENABLE | ||||||
|  |     // This test take a long time to run, disable the WTD until its complete
 | ||||||
|  |     wdt_disable(); | ||||||
|  | #endif | ||||||
|  |     backlight_set(0); | ||||||
|  |     force_issi_refresh(); | ||||||
|  | //    for(uint8_t x = 0; x < sizeof(rgb_sequence); x++){
 | ||||||
|  | //        set_rgb(rgb_sequence[x], 255, 0, 0);
 | ||||||
|  | //        force_issi_refresh();
 | ||||||
|  | //        _delay_ms(250);
 | ||||||
|  | //        set_rgb(rgb_sequence[x], 0, 255, 0);
 | ||||||
|  | //        force_issi_refresh();
 | ||||||
|  | //        _delay_ms(250);
 | ||||||
|  | //        set_rgb(rgb_sequence[x], 0, 0, 255);
 | ||||||
|  | //        force_issi_refresh();
 | ||||||
|  | //        _delay_ms(250);
 | ||||||
|  | //        set_rgb(rgb_sequence[x], 0, 0, 0);
 | ||||||
|  | //        force_issi_refresh();
 | ||||||
|  | //    }
 | ||||||
|  | #ifdef WATCHDOG_ENABLE | ||||||
|  |     wdt_enable(WDTO_250MS); | ||||||
|  | #endif | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void backlight_init_ports(void){ | ||||||
|  |     xprintf("backlight_init_ports\n"); | ||||||
|  |     issi_init(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
							
								
								
									
										9
									
								
								keyboards/meira/lighting.h
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										9
									
								
								keyboards/meira/lighting.h
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,9 @@ | ||||||
|  | #ifndef LIGHTING_H | ||||||
|  | #define LIGHTING_H | ||||||
|  | 
 | ||||||
|  | void led_test(void); | ||||||
|  | void force_issi_refresh(void); | ||||||
|  | void set_backlight(uint8_t level); | ||||||
|  | void set_backlight_by_keymap(uint8_t col, uint8_t row); | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										314
									
								
								keyboards/meira/matrix.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										314
									
								
								keyboards/meira/matrix.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,314 @@ | ||||||
|  | /*
 | ||||||
|  | Copyright 2012 Jun Wako <wakojun@gmail.com> | ||||||
|  | Copyright 2017 Cole Markham <cole@ccmcomputing.net> | ||||||
|  | 
 | ||||||
|  | This program is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 2 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | This program is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | You should have received a copy of the GNU General Public License | ||||||
|  | along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * scan matrix | ||||||
|  |  */ | ||||||
|  | #include <stdint.h> | ||||||
|  | #include <stdbool.h> | ||||||
|  | #if defined(__AVR__) | ||||||
|  | #include <avr/io.h> | ||||||
|  | #endif | ||||||
|  | #include "meira.h" | ||||||
|  | #include "wait.h" | ||||||
|  | #include "print.h" | ||||||
|  | #include "debug.h" | ||||||
|  | #include "util.h" | ||||||
|  | #include "matrix.h" | ||||||
|  | #include "config.h" | ||||||
|  | #include "timer.h" | ||||||
|  | 
 | ||||||
|  | #ifndef DEBOUNCING_DELAY | ||||||
|  | #   define DEBOUNCING_DELAY 5 | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #if (DEBOUNCING_DELAY > 0) | ||||||
|  |     static uint16_t debouncing_time; | ||||||
|  |     static bool debouncing = false; | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #if (MATRIX_COLS <= 8) | ||||||
|  | #    define print_matrix_header()  print("\nr/c 01234567\n") | ||||||
|  | #    define print_matrix_row(row)  print_bin_reverse8(matrix_get_row(row)) | ||||||
|  | #    define matrix_bitpop(i)       bitpop(matrix[i]) | ||||||
|  | #    define ROW_SHIFTER ((uint8_t)1) | ||||||
|  | #elif (MATRIX_COLS <= 16) | ||||||
|  | #    define print_matrix_header()  print("\nr/c 0123456789ABCDEF\n") | ||||||
|  | #    define print_matrix_row(row)  print_bin_reverse16(matrix_get_row(row)) | ||||||
|  | #    define matrix_bitpop(i)       bitpop16(matrix[i]) | ||||||
|  | #    define ROW_SHIFTER ((uint16_t)1) | ||||||
|  | #elif (MATRIX_COLS <= 32) | ||||||
|  | #    define print_matrix_header()  print("\nr/c 0123456789ABCDEF0123456789ABCDEF\n") | ||||||
|  | #    define print_matrix_row(row)  print_bin_reverse32(matrix_get_row(row)) | ||||||
|  | #    define matrix_bitpop(i)       bitpop32(matrix[i]) | ||||||
|  | #    define ROW_SHIFTER  ((uint32_t)1) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | static matrix_row_t matrix_debouncing[MATRIX_ROWS]; | ||||||
|  | 
 | ||||||
|  | static const uint8_t row_pins[MATRIX_ROWS] = MATRIX_ROW_PINS; | ||||||
|  | static const uint8_t col_pins[4] = MATRIX_COL_PINS; | ||||||
|  | //static const uint8_t lrow_pins[MATRIX_ROWS] = LED_ROW_PINS;
 | ||||||
|  | //static const uint8_t lcol_pins[4] = LED_COL_PINS;
 | ||||||
|  | 
 | ||||||
|  | /* matrix state(1:on, 0:off) */ | ||||||
|  | static matrix_row_t matrix[MATRIX_ROWS]; | ||||||
|  | static matrix_row_t matrix_debouncing[MATRIX_ROWS]; | ||||||
|  | static void init_rows(void); | ||||||
|  | //static void init_lcols(void);
 | ||||||
|  | static bool read_rows_on_col(matrix_row_t current_matrix[], uint8_t current_col); | ||||||
|  | static void unselect_cols(void); | ||||||
|  | static void select_col(uint8_t col); | ||||||
|  | 
 | ||||||
|  | __attribute__ ((weak)) | ||||||
|  | void matrix_init_quantum(void) { | ||||||
|  |     matrix_init_kb(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | __attribute__ ((weak)) | ||||||
|  | void matrix_scan_quantum(void) { | ||||||
|  |     matrix_scan_kb(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | __attribute__ ((weak)) | ||||||
|  | void matrix_init_kb(void) { | ||||||
|  |     matrix_init_user(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | __attribute__ ((weak)) | ||||||
|  | void matrix_scan_kb(void) { | ||||||
|  |     matrix_scan_user(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | __attribute__ ((weak)) | ||||||
|  | void matrix_init_user(void) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | __attribute__ ((weak)) | ||||||
|  | void matrix_scan_user(void) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | inline | ||||||
|  | uint8_t matrix_rows(void) | ||||||
|  | { | ||||||
|  |     return MATRIX_ROWS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | inline | ||||||
|  | uint8_t matrix_cols(void) | ||||||
|  | { | ||||||
|  |     return MATRIX_COLS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void matrix_init(void) | ||||||
|  | { | ||||||
|  |     debug_enable = true; | ||||||
|  |     debug_matrix = true; | ||||||
|  |     debug_mouse = true; | ||||||
|  |     // initialize row and col
 | ||||||
|  |     unselect_cols(); | ||||||
|  |     init_rows(); | ||||||
|  | //    init_lcols();
 | ||||||
|  | 
 | ||||||
|  | //    TX_RX_LED_INIT;
 | ||||||
|  | 
 | ||||||
|  |     // initialize matrix state: all keys off
 | ||||||
|  |     for (uint8_t i=0; i < MATRIX_ROWS; i++) { | ||||||
|  |         matrix[i] = 0; | ||||||
|  |         matrix_debouncing[i] = 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     matrix_init_quantum(); | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint8_t _matrix_scan(void) | ||||||
|  | { | ||||||
|  |     // Set col, read rows
 | ||||||
|  |     for (uint8_t current_col = 0; current_col < MATRIX_COLS; current_col++) { | ||||||
|  | #       if (DEBOUNCING_DELAY > 0) | ||||||
|  |             bool matrix_changed = read_rows_on_col(matrix_debouncing, current_col); | ||||||
|  |             if (matrix_changed) { | ||||||
|  |                 debouncing = true; | ||||||
|  |                 debouncing_time = timer_read(); | ||||||
|  |             } | ||||||
|  | #       else | ||||||
|  |              read_rows_on_col(matrix, current_col); | ||||||
|  | #       endif | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | #   if (DEBOUNCING_DELAY > 0) | ||||||
|  |         if (debouncing && (timer_elapsed(debouncing_time) > DEBOUNCING_DELAY)) { | ||||||
|  |             for (uint8_t i = 0; i < MATRIX_ROWS; i++) { | ||||||
|  |                 matrix[i] = matrix_debouncing[i]; | ||||||
|  |             } | ||||||
|  |             debouncing = false; | ||||||
|  |         } | ||||||
|  | #   endif | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint8_t matrix_scan(void) | ||||||
|  | { | ||||||
|  | 	uint8_t ret = _matrix_scan(); | ||||||
|  | 	matrix_scan_quantum(); | ||||||
|  | //	// HACK backlighting
 | ||||||
|  | //	for (uint8_t t = 0; t < meira_get_backlight_level(); t++) {
 | ||||||
|  | //		for (uint8_t x = 0; x < 13; x++) {
 | ||||||
|  | //			for (uint8_t y = 0; y < 4; y++) {
 | ||||||
|  | //				uint8_t pin = lcol_pins[y];
 | ||||||
|  | //				if ((x >> y) & 1) {
 | ||||||
|  | //					_SFR_IO8((pin >> 4) + 2) |=  _BV(pin & 0xF); // HI
 | ||||||
|  | //				} else {
 | ||||||
|  | //					_SFR_IO8((pin >> 4) + 2) &=  ~_BV(pin & 0xF); // LO
 | ||||||
|  | //				}
 | ||||||
|  | //			}
 | ||||||
|  | //		}
 | ||||||
|  | //	}
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool matrix_is_modified(void) | ||||||
|  | { | ||||||
|  |     if (debouncing) return false; | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | inline | ||||||
|  | bool matrix_is_on(uint8_t row, uint8_t col) | ||||||
|  | { | ||||||
|  |     return (matrix[row] & ((matrix_row_t)1<<col)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | inline | ||||||
|  | matrix_row_t matrix_get_row(uint8_t row) | ||||||
|  | { | ||||||
|  |     return matrix[row]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void matrix_print(void) | ||||||
|  | { | ||||||
|  |     print("\nr/c 0123456789ABCDEF\n"); | ||||||
|  |     for (uint8_t row = 0; row < MATRIX_ROWS; row++) { | ||||||
|  |         phex(row); print(": "); | ||||||
|  |         pbin_reverse16(matrix_get_row(row)); | ||||||
|  |         print("\n"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint8_t matrix_key_count(void) | ||||||
|  | { | ||||||
|  |     uint8_t count = 0; | ||||||
|  |     for (uint8_t i = 0; i < MATRIX_ROWS; i++) { | ||||||
|  |         count += bitpop16(matrix[i]); | ||||||
|  |     } | ||||||
|  |     return count; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | static void init_rows(void) | ||||||
|  | { | ||||||
|  |     for(uint8_t x = 0; x < MATRIX_ROWS; x++) { | ||||||
|  |         uint8_t pin = row_pins[x]; | ||||||
|  |         _SFR_IO8((pin >> 4) + 1) &= ~_BV(pin & 0xF); // IN
 | ||||||
|  |         _SFR_IO8((pin >> 4) + 2) |=  _BV(pin & 0xF); // HI
 | ||||||
|  | //        // HACK backlighting
 | ||||||
|  | //        uint8_t lpin = lrow_pins[x];
 | ||||||
|  | //        _SFR_IO8((lpin >> 4) + 1) |= _BV(lpin & 0xF); // OUT
 | ||||||
|  | //        _SFR_IO8((lpin >> 4) + 2) |=  _BV(lpin & 0xF); // HI
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | //static void init_lcols(void)
 | ||||||
|  | //{
 | ||||||
|  | //	for (uint8_t x = 0; x < 4; x++) {
 | ||||||
|  | //		uint8_t pin = lcol_pins[x];
 | ||||||
|  | //		_SFR_IO8((pin >> 4) + 1) |= _BV(pin & 0xF); // OUT
 | ||||||
|  | //		_SFR_IO8((pin >> 4) + 2) |= _BV(pin & 0xF); // HIGH
 | ||||||
|  | //	}
 | ||||||
|  | //}
 | ||||||
|  | 
 | ||||||
|  | static bool read_rows_on_col(matrix_row_t current_matrix[], uint8_t current_col) | ||||||
|  | { | ||||||
|  |     bool matrix_changed = false; | ||||||
|  | 
 | ||||||
|  |     // Select col and wait for col selection to stabilize
 | ||||||
|  |     select_col(current_col); | ||||||
|  |     wait_us(30); | ||||||
|  | 
 | ||||||
|  |     // For each row...
 | ||||||
|  |     for(uint8_t row_index = 0; row_index < MATRIX_ROWS; row_index++) | ||||||
|  |     { | ||||||
|  | 
 | ||||||
|  |         // Store last value of row prior to reading
 | ||||||
|  |         matrix_row_t last_row_value = current_matrix[row_index]; | ||||||
|  | 
 | ||||||
|  |         // Check row pin state
 | ||||||
|  |         if ((_SFR_IO8(row_pins[row_index] >> 4) & _BV(row_pins[row_index] & 0xF)) == 0) | ||||||
|  |         { | ||||||
|  |             // Pin LO, set col bit
 | ||||||
|  |             current_matrix[row_index] |= (ROW_SHIFTER << current_col); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             // Pin HI, clear col bit
 | ||||||
|  |             current_matrix[row_index] &= ~(ROW_SHIFTER << current_col); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Determine if the matrix changed state
 | ||||||
|  |         if ((last_row_value != current_matrix[row_index]) && !(matrix_changed)) | ||||||
|  |         { | ||||||
|  |             matrix_changed = true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Unselect col
 | ||||||
|  |     unselect_cols(); | ||||||
|  | 
 | ||||||
|  |     return matrix_changed; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void select_col(uint8_t col) | ||||||
|  | { | ||||||
|  | #ifdef FLIPPED_BOARD | ||||||
|  | 	col = MATRIX_COLS - col - 1; | ||||||
|  | #endif | ||||||
|  |     for(uint8_t x = 0; x < 4; x++) { | ||||||
|  | 		uint8_t pin = col_pins[x]; | ||||||
|  |         _SFR_IO8((pin >> 4) + 1) |= _BV(pin & 0xF); // OUT
 | ||||||
|  | 		if (((col >> x) & 0x1) == 1){ | ||||||
|  | 			_SFR_IO8((pin >> 4) + 2) |=  _BV(pin & 0xF); // HIGH
 | ||||||
|  | 		} else { | ||||||
|  | 			_SFR_IO8((pin >> 4) + 2) &=  ~_BV(pin & 0xF); // LOW
 | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void unselect_cols(void) | ||||||
|  | { | ||||||
|  | 	// FIXME This really needs to use the global enable on the decoder, because currently this sets the value to col1
 | ||||||
|  |     for(uint8_t x = 0; x < 4; x++) { | ||||||
|  |         uint8_t pin = col_pins[x]; | ||||||
|  |         _SFR_IO8((pin >> 4) + 1) |= _BV(pin & 0xF); // OUT
 | ||||||
|  |         _SFR_IO8((pin >> 4) + 2) &=  ~_BV(pin & 0xF); // LOW
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
							
								
								
									
										145
									
								
								keyboards/meira/meira.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								keyboards/meira/meira.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,145 @@ | ||||||
|  | /* Copyright 2017 REPLACE_WITH_YOUR_NAME
 | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  |  */ | ||||||
|  | #include "meira.h" | ||||||
|  | #include "issi.h" | ||||||
|  | #include "TWIlib.h" | ||||||
|  | #include "lighting.h" | ||||||
|  | #include "quantum.h" | ||||||
|  | #define BACKLIGHT_BREATHING | ||||||
|  | 
 | ||||||
|  | #ifdef AUDIO_ENABLE | ||||||
|  |     float tone_startup[][2] = SONG(STARTUP_SOUND); | ||||||
|  |     float tone_goodbye[][2] = SONG(GOODBYE_SOUND); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void shutdown_user(void) { | ||||||
|  |     #ifdef AUDIO_ENABLE | ||||||
|  |         PLAY_NOTE_ARRAY(tone_goodbye, false, 0); | ||||||
|  |     _delay_ms(150); | ||||||
|  |     stop_all_notes(); | ||||||
|  |     #endif | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void matrix_init_kb(void) | ||||||
|  | { | ||||||
|  |     debug_enable=true; | ||||||
|  |     print("meira matrix_init_kb\n"); | ||||||
|  | #ifdef AUDIO_ENABLE | ||||||
|  |     _delay_ms(20); // gets rid of tick
 | ||||||
|  |     PLAY_NOTE_ARRAY(tone_startup, false, 0); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #ifdef ISSI_ENABLE | ||||||
|  |     issi_init(); | ||||||
|  | #endif | ||||||
|  |     backlight_set(5); | ||||||
|  | #ifdef WATCHDOG_ENABLE | ||||||
|  |     // This is done after turning the layer LED red, if we're caught in a loop
 | ||||||
|  |     // we should get a flashing red light
 | ||||||
|  |     wdt_enable(WDTO_500MS); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     // put your keyboard start-up code here
 | ||||||
|  |     // runs once when the firmware starts up
 | ||||||
|  |     matrix_init_user(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void matrix_scan_kb(void) | ||||||
|  | { | ||||||
|  | #ifdef WATCHDOG_ENABLE | ||||||
|  |     wdt_reset(); | ||||||
|  | #endif | ||||||
|  | #ifdef ISSI_ENABLE | ||||||
|  |     // switch/underglow lighting update
 | ||||||
|  |     static uint32_t issi_device = 0; | ||||||
|  |     static uint32_t twi_last_ready = 0; | ||||||
|  |     if(twi_last_ready > 1000){ | ||||||
|  |         // Its been way too long since the last ISSI update, reset the I2C bus and start again
 | ||||||
|  |         xprintf("TWI failed to recover, TWI re-init\n"); | ||||||
|  |         twi_last_ready = 0; | ||||||
|  |         TWIInit(); | ||||||
|  |         force_issi_refresh(); | ||||||
|  |     } | ||||||
|  |     if(isTWIReady()){ | ||||||
|  |         twi_last_ready = 0; | ||||||
|  |         // If the i2c bus is available, kick off the issi update, alternate between devices
 | ||||||
|  |         update_issi(issi_device, issi_device); | ||||||
|  |         if(issi_device){ | ||||||
|  |             issi_device = 0; | ||||||
|  |         }else{ | ||||||
|  |             issi_device = 3; | ||||||
|  |         } | ||||||
|  |     }else{ | ||||||
|  |         twi_last_ready++; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  |     matrix_scan_user(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool process_record_kb(uint16_t keycode, keyrecord_t *record) { | ||||||
|  |     // Test code that turns on the switch led for the key that is pressed
 | ||||||
|  |     // set_backlight_by_keymap(record->event.key.col, record->event.key.row);
 | ||||||
|  |     if (keycode == RESET) { | ||||||
|  |         reset_keyboard_kb(); | ||||||
|  |     } else { | ||||||
|  |     } | ||||||
|  | 	return process_record_user(keycode, record); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void led_set_kb(uint8_t usb_led) { | ||||||
|  | 	// put your keyboard LED indicator (ex: Caps Lock LED) toggling code here
 | ||||||
|  | 	led_set_user(usb_led); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | //void action_function(keyrecord_t *event, uint8_t id, uint8_t opt)
 | ||||||
|  | //{
 | ||||||
|  | //#ifdef AUDIO_ENABLE
 | ||||||
|  | //    int8_t sign = 1;
 | ||||||
|  | //#endif
 | ||||||
|  | //    if(id == LFK_ESC_TILDE){
 | ||||||
|  | //        // Send ~ on shift-esc
 | ||||||
|  | //        void (*method)(uint8_t) = (event->event.pressed) ? &add_key : &del_key;
 | ||||||
|  | //        uint8_t shifted = get_mods() & (MOD_BIT(KC_LSHIFT) | MOD_BIT(KC_RSHIFT));
 | ||||||
|  | //        method(shifted ? KC_GRAVE : KC_ESCAPE);
 | ||||||
|  | //        send_keyboard_report();
 | ||||||
|  | //    }else if(event->event.pressed){
 | ||||||
|  | //        switch(id){
 | ||||||
|  | //            case LFK_CLEAR:
 | ||||||
|  | //                // Go back to default layer
 | ||||||
|  | //                layer_clear();
 | ||||||
|  | //                break;
 | ||||||
|  | //#ifdef ISSI_ENABLE
 | ||||||
|  | //            case LFK_LED_TEST:
 | ||||||
|  | //                led_test();
 | ||||||
|  | //                break;
 | ||||||
|  | //#endif
 | ||||||
|  | //        }
 | ||||||
|  | //    }
 | ||||||
|  | //}
 | ||||||
|  | 
 | ||||||
|  | void reset_keyboard_kb(){ | ||||||
|  | #ifdef WATCHDOG_ENABLE | ||||||
|  |     MCUSR = 0; | ||||||
|  |     wdt_disable(); | ||||||
|  |     wdt_reset(); | ||||||
|  | #endif | ||||||
|  |     xprintf("programming!\n"); | ||||||
|  |     reset_keyboard(); | ||||||
|  | } | ||||||
							
								
								
									
										48
									
								
								keyboards/meira/meira.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								keyboards/meira/meira.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,48 @@ | ||||||
|  | /* Copyright 2017 Cole Markham
 | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 2 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  |  */ | ||||||
|  | #ifndef MEIRA_H | ||||||
|  | #define MEIRA_H | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #ifdef SUBPROJECT_featherble | ||||||
|  |     #include "featherble.h" | ||||||
|  | #endif | ||||||
|  | #ifdef SUBPROJECT_promicro | ||||||
|  |     #include "promicro.h" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #include "quantum.h" | ||||||
|  | 
 | ||||||
|  | void reset_keyboard_kb(void); | ||||||
|  | 
 | ||||||
|  | // This a shortcut to help you visually see your layout.
 | ||||||
|  | // The following is an example using the Planck MIT layout
 | ||||||
|  | // The first section contains all of the arguments
 | ||||||
|  | // The second converts the arguments into a two-dimensional array
 | ||||||
|  | #define KEYMAP( \ | ||||||
|  |    k00, k01, k02, k03, k04, k05, k06, k07, k08, k09, k0a, k0b, \ | ||||||
|  | 	k10, k11, k12, k13, k14, k15, k16, k17, k18, k19, k1a, k1b, \ | ||||||
|  | 	k20, k21, k22, k23, k24, k25, k26, k27, k28, k29, k2a, k2b, \ | ||||||
|  | 	k30, k31, k32, k33, k34, k35, k36, k37, k38, k39, k3a, k3b \ | ||||||
|  | ) \ | ||||||
|  | { \ | ||||||
|  | 	{ k00, k01, k02, k03, k04, k05, k06, k07, k08, k09, k0a, k0b }, \ | ||||||
|  | 	{ k10, k11, k12, k13, k14, k15, k16, k17, k18, k19, k1a, k1b }, \ | ||||||
|  | 	{ k20, k21, k22, k23, k24, k25, k26, k27, k28, k29, k2a, k2b }, \ | ||||||
|  | 	{ k30, k31, k32, k33, k34, k35, k36, k37, k38, k39, k3a, k3b } \ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										168
									
								
								keyboards/meira/promicro/config.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								keyboards/meira/promicro/config.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,168 @@ | ||||||
|  | /*
 | ||||||
|  | Copyright 2017 REPLACE_WITH_YOUR_NAME | ||||||
|  | 
 | ||||||
|  | This program is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 2 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | This program is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | You should have received a copy of the GNU General Public License | ||||||
|  | along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | #ifndef PROMICROCONFIG_H | ||||||
|  | #define PROMICROCONFIG_H | ||||||
|  | 
 | ||||||
|  | #include "config_common.h" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Keyboard Matrix Assignments | ||||||
|  |  * | ||||||
|  |  * Change this to how you wired your keyboard | ||||||
|  |  * COLS: AVR pins used for columns, left to right | ||||||
|  |  * ROWS: AVR pins used for rows, top to bottom | ||||||
|  |  * DIODE_DIRECTION: COL2ROW = COL = Anode (+), ROW = Cathode (-, marked on diode) | ||||||
|  |  *                  ROW2COL = ROW = Anode (+), COL = Cathode (-, marked on diode) | ||||||
|  |  * | ||||||
|  | */ | ||||||
|  | #define MATRIX_ROW_PINS { F7, F6, F5, F4 } | ||||||
|  | // Column pins to demux in LSB order
 | ||||||
|  | #define MATRIX_COL_PINS { B1, B3, B2, B6 } | ||||||
|  | #define LED_EN_PIN D2 | ||||||
|  | #define UNUSED_PINS | ||||||
|  | 
 | ||||||
|  | #define CATERINA_BOOTLOADER | ||||||
|  | 
 | ||||||
|  | /* Debounce reduces chatter (unintended double-presses) - set 0 if debouncing is not needed */ | ||||||
|  | #define DEBOUNCING_DELAY 5 | ||||||
|  | 
 | ||||||
|  | /* define if matrix has ghost (lacks anti-ghosting diodes) */ | ||||||
|  | //#define MATRIX_HAS_GHOST
 | ||||||
|  | 
 | ||||||
|  | /* Mechanical locking support. Use KC_LCAP, KC_LNUM or KC_LSCR instead in keymap */ | ||||||
|  | //#define LOCKING_SUPPORT_ENABLE
 | ||||||
|  | /* Locking resynchronize hack */ | ||||||
|  | //#define LOCKING_RESYNC_ENABLE
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Force NKRO | ||||||
|  |  * | ||||||
|  |  * Force NKRO (nKey Rollover) to be enabled by default, regardless of the saved | ||||||
|  |  * state in the bootmagic EEPROM settings. (Note that NKRO must be enabled in the | ||||||
|  |  * makefile for this to work.) | ||||||
|  |  * | ||||||
|  |  * If forced on, NKRO can be disabled via magic key (default = LShift+RShift+N) | ||||||
|  |  * until the next keyboard reset. | ||||||
|  |  * | ||||||
|  |  * NKRO may prevent your keystrokes from being detected in the BIOS, but it is | ||||||
|  |  * fully operational during normal computer usage. | ||||||
|  |  * | ||||||
|  |  * For a less heavy-handed approach, enable NKRO via magic key (LShift+RShift+N) | ||||||
|  |  * or via bootmagic (hold SPACE+N while plugging in the keyboard). Once set by | ||||||
|  |  * bootmagic, NKRO mode will always be enabled until it is toggled again during a | ||||||
|  |  * power-up. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | //#define FORCE_NKRO
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Magic Key Options | ||||||
|  |  * | ||||||
|  |  * Magic keys are hotkey commands that allow control over firmware functions of | ||||||
|  |  * the keyboard. They are best used in combination with the HID Listen program, | ||||||
|  |  * found here: https://www.pjrc.com/teensy/hid_listen.html
 | ||||||
|  |  * | ||||||
|  |  * The options below allow the magic key functionality to be changed. This is | ||||||
|  |  * useful if your keyboard/keypad is missing keys and you want magic key support. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* key combination for magic key command */ | ||||||
|  | #define IS_COMMAND() ( \ | ||||||
|  |     keyboard_report->mods == (MOD_BIT(KC_LSHIFT) | MOD_BIT(KC_RSHIFT)) \ | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | /* control how magic key switches layers */ | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_FKEYS  true
 | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_NKEYS  true
 | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_CUSTOM false
 | ||||||
|  | 
 | ||||||
|  | /* override magic key keymap */ | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_FKEYS
 | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_NKEYS
 | ||||||
|  | //#define MAGIC_KEY_SWITCH_LAYER_WITH_CUSTOM
 | ||||||
|  | //#define MAGIC_KEY_HELP1          H
 | ||||||
|  | //#define MAGIC_KEY_HELP2          SLASH
 | ||||||
|  | //#define MAGIC_KEY_DEBUG          D
 | ||||||
|  | //#define MAGIC_KEY_DEBUG_MATRIX   X
 | ||||||
|  | //#define MAGIC_KEY_DEBUG_KBD      K
 | ||||||
|  | //#define MAGIC_KEY_DEBUG_MOUSE    M
 | ||||||
|  | //#define MAGIC_KEY_VERSION        V
 | ||||||
|  | //#define MAGIC_KEY_STATUS         S
 | ||||||
|  | //#define MAGIC_KEY_CONSOLE        C
 | ||||||
|  | //#define MAGIC_KEY_LAYER0_ALT1    ESC
 | ||||||
|  | //#define MAGIC_KEY_LAYER0_ALT2    GRAVE
 | ||||||
|  | //#define MAGIC_KEY_LAYER0         0
 | ||||||
|  | //#define MAGIC_KEY_LAYER1         1
 | ||||||
|  | //#define MAGIC_KEY_LAYER2         2
 | ||||||
|  | //#define MAGIC_KEY_LAYER3         3
 | ||||||
|  | //#define MAGIC_KEY_LAYER4         4
 | ||||||
|  | //#define MAGIC_KEY_LAYER5         5
 | ||||||
|  | //#define MAGIC_KEY_LAYER6         6
 | ||||||
|  | //#define MAGIC_KEY_LAYER7         7
 | ||||||
|  | //#define MAGIC_KEY_LAYER8         8
 | ||||||
|  | //#define MAGIC_KEY_LAYER9         9
 | ||||||
|  | //#define MAGIC_KEY_BOOTLOADER     PAUSE
 | ||||||
|  | //#define MAGIC_KEY_LOCK           CAPS
 | ||||||
|  | //#define MAGIC_KEY_EEPROM         E
 | ||||||
|  | //#define MAGIC_KEY_NKRO           N
 | ||||||
|  | //#define MAGIC_KEY_SLEEP_LED      Z
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Feature disable options | ||||||
|  |  *  These options are also useful to firmware size reduction. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* disable debug print */ | ||||||
|  | //#define NO_DEBUG
 | ||||||
|  | 
 | ||||||
|  | /* disable print */ | ||||||
|  | //#define NO_PRINT
 | ||||||
|  | 
 | ||||||
|  | /* disable action features */ | ||||||
|  | //#define NO_ACTION_LAYER
 | ||||||
|  | //#define NO_ACTION_TAPPING
 | ||||||
|  | //#define NO_ACTION_ONESHOT
 | ||||||
|  | //#define NO_ACTION_MACRO
 | ||||||
|  | //#define NO_ACTION_FUNCTION
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * MIDI options | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* Prevent use of disabled MIDI features in the keymap */ | ||||||
|  | //#define MIDI_ENABLE_STRICT 1
 | ||||||
|  | 
 | ||||||
|  | /* enable basic MIDI features:
 | ||||||
|  |    - MIDI notes can be sent when in Music mode is on | ||||||
|  | */ | ||||||
|  | //#define MIDI_BASIC
 | ||||||
|  | 
 | ||||||
|  | /* enable advanced MIDI features:
 | ||||||
|  |    - MIDI notes can be added to the keymap | ||||||
|  |    - Octave shift and transpose | ||||||
|  |    - Virtual sustain, portamento, and modulation wheel | ||||||
|  |    - etc. | ||||||
|  | */ | ||||||
|  | //#define MIDI_ADVANCED
 | ||||||
|  | 
 | ||||||
|  | /* override number of MIDI tone keycodes (each octave adds 12 keycodes and allocates 12 bytes) */ | ||||||
|  | //#define MIDI_TONE_KEYCODE_OCTAVES 1
 | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										2
									
								
								keyboards/meira/promicro/promicro.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								keyboards/meira/promicro/promicro.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | #include "meira.h" | ||||||
|  | 
 | ||||||
							
								
								
									
										10
									
								
								keyboards/meira/promicro/promicro.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								keyboards/meira/promicro/promicro.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | ||||||
|  | #ifndef FEATHERBLE_H | ||||||
|  | #define FEATHERBLE_H | ||||||
|  | 
 | ||||||
|  | #include "../meira.h" | ||||||
|  | 
 | ||||||
|  | #include "quantum.h" | ||||||
|  | 
 | ||||||
|  | #include "pro_micro.h" | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										2
									
								
								keyboards/meira/promicro/rules.mk
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								keyboards/meira/promicro/rules.mk
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | BLUETOOTH_ENABLE = no | ||||||
|  | BACKLIGHT_ENABLE = yes | ||||||
							
								
								
									
										28
									
								
								keyboards/meira/readme.md
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								keyboards/meira/readme.md
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | ||||||
|  | meira keyboard firmware | ||||||
|  | ====================== | ||||||
|  | 
 | ||||||
|  | ## Quantum MK Firmware | ||||||
|  | 
 | ||||||
|  | For the full Quantum feature list, see [the parent readme](/). | ||||||
|  | 
 | ||||||
|  | ## Building | ||||||
|  | 
 | ||||||
|  | Download or clone the whole firmware and navigate to the keyboards/meira folder. Once your dev env is setup, you'll be able to type `make` to generate your .hex - you can then use the Teensy Loader to program your .hex file.  | ||||||
|  | 
 | ||||||
|  | Depending on which keymap you would like to use, you will have to compile slightly differently. | ||||||
|  | 
 | ||||||
|  | ### Default | ||||||
|  | 
 | ||||||
|  | To build with the default keymap, simply run `make default`. | ||||||
|  | 
 | ||||||
|  | ### Other Keymaps | ||||||
|  | 
 | ||||||
|  | Several version of keymap are available in advance but you are recommended to define your favorite layout yourself. To define your own keymap create a folder with the name of your keymap in the keymaps folder, and see keymap documentation (you can find in top readme.md) and existant keymap files. | ||||||
|  | 
 | ||||||
|  | To build the firmware binary hex file with a keymap just do `make` with a keymap like this: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | $ make [default|jack|<name>] | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | Keymaps follow the format **__\<name\>.c__** and are stored in the `keymaps` folder. | ||||||
							
								
								
									
										84
									
								
								keyboards/meira/rules.mk
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								keyboards/meira/rules.mk
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,84 @@ | ||||||
|  | SRC += matrix.c TWIlib.c issi.c lighting.c | ||||||
|  | 
 | ||||||
|  | # MCU name
 | ||||||
|  | #MCU = at90usb1286
 | ||||||
|  | MCU = atmega32u4 | ||||||
|  | 
 | ||||||
|  | # Processor frequency.
 | ||||||
|  | #     This will define a symbol, F_CPU, in all source code files equal to the
 | ||||||
|  | #     processor frequency in Hz. You can then use this symbol in your source code to
 | ||||||
|  | #     calculate timings. Do NOT tack on a 'UL' at the end, this will be done
 | ||||||
|  | #     automatically to create a 32-bit value in your source code.
 | ||||||
|  | #
 | ||||||
|  | #     This will be an integer division of F_USB below, as it is sourced by
 | ||||||
|  | #     F_USB after it has run through any CPU prescalers. Note that this value
 | ||||||
|  | #     does not *change* the processor frequency - it should merely be updated to
 | ||||||
|  | #     reflect the processor speed set externally so that the code can use accurate
 | ||||||
|  | #     software delays.
 | ||||||
|  | F_CPU = 16000000 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #
 | ||||||
|  | # LUFA specific
 | ||||||
|  | #
 | ||||||
|  | # Target architecture (see library "Board Types" documentation).
 | ||||||
|  | ARCH = AVR8 | ||||||
|  | 
 | ||||||
|  | # Input clock frequency.
 | ||||||
|  | #     This will define a symbol, F_USB, in all source code files equal to the
 | ||||||
|  | #     input clock frequency (before any prescaling is performed) in Hz. This value may
 | ||||||
|  | #     differ from F_CPU if prescaling is used on the latter, and is required as the
 | ||||||
|  | #     raw input clock is fed directly to the PLL sections of the AVR for high speed
 | ||||||
|  | #     clock generation for the USB and other AVR subsections. Do NOT tack on a 'UL'
 | ||||||
|  | #     at the end, this will be done automatically to create a 32-bit value in your
 | ||||||
|  | #     source code.
 | ||||||
|  | #
 | ||||||
|  | #     If no clock division is performed on the input clock inside the AVR (via the
 | ||||||
|  | #     CPU clock adjust registers or the clock division fuses), this will be equal to F_CPU.
 | ||||||
|  | F_USB = $(F_CPU) | ||||||
|  | 
 | ||||||
|  | # Interrupt driven control endpoint task(+60)
 | ||||||
|  | OPT_DEFS += -DINTERRUPT_CONTROL_ENDPOINT | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Boot Section Size in *bytes*
 | ||||||
|  | #   Teensy halfKay   512
 | ||||||
|  | #   Teensy++ halfKay 1024
 | ||||||
|  | #   Atmel DFU loader 4096
 | ||||||
|  | #   LUFA bootloader  4096
 | ||||||
|  | #   USBaspLoader     2048
 | ||||||
|  | OPT_DEFS += -DBOOTLOADER_SIZE=512 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Build Options
 | ||||||
|  | #   change yes to no to disable
 | ||||||
|  | #
 | ||||||
|  | BOOTMAGIC_ENABLE ?= no      # Virtual DIP switch configuration(+1000) | ||||||
|  | MOUSEKEY_ENABLE ?= yes       # Mouse keys(+4700) | ||||||
|  | EXTRAKEY_ENABLE ?= yes       # Audio control and System control(+450) | ||||||
|  | CONSOLE_ENABLE ?= yes        # Console for debug(+400) | ||||||
|  | COMMAND_ENABLE ?= yes        # Commands for debug and configuration | ||||||
|  | # Do not enable SLEEP_LED_ENABLE. it uses the same timer as BACKLIGHT_ENABLE
 | ||||||
|  | SLEEP_LED_ENABLE ?= no       # Breathing sleep LED during USB suspend | ||||||
|  | # if this doesn't work, see here: https://github.com/tmk/tmk_keyboard/wiki/FAQ#nkro-doesnt-work
 | ||||||
|  | NKRO_ENABLE ?= no            # USB Nkey Rollover | ||||||
|  | BACKLIGHT_ENABLE = yes       # Enable keyboard backlight functionality on B7 by default | ||||||
|  | MIDI_ENABLE ?= no            # MIDI support (+2400 to 4200, depending on config) | ||||||
|  | UNICODE_ENABLE ?= no         # Unicode | ||||||
|  | BLUETOOTH_ENABLE ?= no       # Enable Bluetooth with the Adafruit EZ-Key HID | ||||||
|  | AUDIO_ENABLE ?= no           # Audio output on port C6 | ||||||
|  | RGBLIGHT_ENABLE ?= no       # Enable WS2812 RGB underlight.  Do not enable this with audio at the same time. | ||||||
|  | FAUXCLICKY_ENABLE ?= no      # Use buzzer to emulate clicky switches | ||||||
|  | 
 | ||||||
|  | ISSI_ENABLE = yes			# If the I2C pullup resistors aren't install this must be disabled | ||||||
|  | #WATCHDOG_ENABLE = yes		# Resets keyboard if matrix_scan isn't run every 250ms
 | ||||||
|  | 
 | ||||||
|  | CUSTOM_MATRIX = yes | ||||||
|  | 
 | ||||||
|  | ifeq ($(strip $(ISSI_ENABLE)), yes) | ||||||
|  |     TMK_COMMON_DEFS += -DISSI_ENABLE | ||||||
|  | endif | ||||||
|  | 
 | ||||||
|  | ifeq ($(strip $(WATCHDOG_ENABLE)), yes) | ||||||
|  |     TMK_COMMON_DEFS += -DWATCHDOG_ENABLE | ||||||
|  | endif | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Cole Markham
						Cole Markham