aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--keyboards/jj4x4/README.md71
-rw-r--r--keyboards/jj4x4/backlight.c213
-rw-r--r--keyboards/jj4x4/backlight_custom.h15
-rw-r--r--keyboards/jj4x4/breathing_custom.h140
-rw-r--r--keyboards/jj4x4/config.h53
-rw-r--r--keyboards/jj4x4/i2c.c104
-rw-r--r--keyboards/jj4x4/i2c.h25
-rw-r--r--keyboards/jj4x4/info.json23
-rw-r--r--keyboards/jj4x4/jj4x4.c97
-rw-r--r--keyboards/jj4x4/jj4x4.h46
-rw-r--r--keyboards/jj4x4/keymaps/default/keymap.c34
-rw-r--r--keyboards/jj4x4/rules.mk58
-rw-r--r--keyboards/jj4x4/tools/README.md16
-rw-r--r--keyboards/jj4x4/tools/usb_detach.c33
-rw-r--r--keyboards/jj4x4/usbconfig.h397
15 files changed, 1325 insertions, 0 deletions
diff --git a/keyboards/jj4x4/README.md b/keyboards/jj4x4/README.md
new file mode 100644
index 000000000..09684bd82
--- /dev/null
+++ b/keyboards/jj4x4/README.md
@@ -0,0 +1,71 @@
1# jj4x4
2
3![jj4x4](https://cdn.shopify.com/s/files/1/2711/4238/products/JJ4x4case-1_1024x1024.jpg?v=1532325339)
4
5A 4x4 keypad kit made and KPRepublic on AliExpress. This is a chopped off version of the jj40 with rearranged keys.
6
7Keyboard Maintainer: [QMK Community](https://github.com/qmk)
8Hardware Supported: Atmega32A
9Hardware Availability: [AliExpress](https://www.aliexpress.com/item/jj4x4-jj4X4-16-keys-Custom-Mechanical-Keyboard-PCB-programmed-numpad-layouts-bface-firmware-with-rgb-bottom/32901955446.html?spm=2114.search0104.3.7.3ebf431ae1d9ic&ws_ab_test=searchweb0_0,searchweb201602_4_10065_10130_10068_10547_319_317_10548_10545_10696_453_10084_454_10083_433_10618_431_10307_537_536_10902_10059_10884_10887_321_322_10103,searchweb201603_6,ppcSwitch_0&algo_expid=9d1891dd-80af-4793-a889-5a62e1fdfdd8-1&algo_pvid=9d1891dd-80af-4793-a889-5a62e1fdfdd8&transAbTest=ae803_5)
10
11Make example for this keyboard (after setting up your build environment):
12
13 make jj4x4:default:program
14
15See [build environment setup](https://docs.qmk.fm/#/getting_started_build_tools) then the [make instructions](https://docs.qmk.fm/#/getting_started_make_guide) for more information.
16
17Note that this is a complete replacement for the firmware, so you won't be
18using Bootmapper Client to change any keyboard settings, since not all the
19USB report options are supported.
20
21In addition you may need the AVR toolchain and `bootloadHID` ([GitHub repo](https://github.com/whiteneon/bootloadHID)) for flashing:
22
23For macOS:
24```
25$ brew cask install crosspack-avr
26$ brew install --HEAD https://raw.githubusercontent.com/robertgzr/homebrew-tap/master/bootloadhid.rb
27```
28
29For Linux:
30```
31$ sudo apt install libusb-dev
32$ wget https://www.obdev.at/downloads/vusb/bootloadHID.2012-12-08.tar.gz
33$ tar -xzf bootloadHID.2012-12-08.tar.gz
34$ cd bootloadHID.2012-12-08/commandline
35$ make
36$ sudo cp bootloadHID /usr/bin
37```
38
39In order to use the `./program` script, which can reboot the board into
40the bootloader, you'll need Python 2 with PyUSB installed:
41
42```
43$ pip install pyusb
44```
45
46If you prefer (or are having issues with a `program` flash), you can just build it (`make jj40:<keymap-name>` and flash the firmware (`.hex` file) directly with
47`bootloadHID` if you boot the board while holding down `8` (second from top, second from left, with usb plug is at the top) to keep it
48in the bootloader:
49
50```
51$ make jj40
52$ bootloadHID -r jj4x4_default.hex
53```
54
55For Windows 10:
56Windows sometimes doesn't recognize the jj4x4. The easiest way of flashing a new layout is probably using [HIDBootFlash](http://vusb.wikidot.com/project:hidbootflash).
571. Go to Windows Device Manager and find the keyboard (plug it in while holding down `8` (second from top, second from left, with usb plug is at the top)). It can be found under Human Interface Devices or under Keyboards.
582. Go to properties and the Details tab to find the hardware ID. You want the VID and the PID (code after the underscore). Plug them into HIDBootFlash and hit Find Device.
593. Use `make jj4x4:<keymap-name>` to generate the .hex file in the qmk basis folder. Select the .hex file in HIDBootFlash and press Flash Device.
60
61
62## Troubleshooting
63
641. Try plugging the board in while pressing `8` (usb plug at top, second from top, second from left). This will force it
65 to boot only the bootloader without loading the firmware. Once this is
66 done, just reflash the board with the original firmware.
672. Sometimes USB hubs can act weird, so try connecting the board directly
68 to your computer or plugging/unplugging the USB hub.
693. If you get an error such as "Resource Unavailable" when attemting to flash
70 on Linux, you may want to compile and run `tools/usb_detach.c`. See `tools/README.md`
71 for more info.
diff --git a/keyboards/jj4x4/backlight.c b/keyboards/jj4x4/backlight.c
new file mode 100644
index 000000000..fbd241fa9
--- /dev/null
+++ b/keyboards/jj4x4/backlight.c
@@ -0,0 +1,213 @@
1/**
2 * Backlighting code for PS2AVRGB boards (ATMEGA32A)
3 * Kenneth A. (github.com/krusli | krusli.me)
4 */
5
6#include "backlight.h"
7#include "quantum.h"
8
9#include <avr/pgmspace.h>
10#include <avr/interrupt.h>
11
12#include "backlight_custom.h"
13#include "breathing_custom.h"
14
15// DEBUG
16#include <stdlib.h>
17#include <stdio.h>
18
19// Port D: digital pins of the AVR chipset
20#define NUMLOCK_PORT (1 << 0) // D0
21#define CAPSLOCK_PORT (1 << 1) // D1
22#define BACKLIGHT_PORT (1 << 4) // D4
23#define SCROLLLOCK_PORT (1 << 6) // D6
24
25#define TIMER_CLK_DIV64 0x03 ///< Timer clocked at F_CPU/64
26#define TIMER1PRESCALE TIMER_CLK_DIV64 ///< timer 1 prescaler default
27
28#define TIMER_PRESCALE_MASK 0x07 ///< Timer Prescaler Bit-Mask
29
30#define PWM_MAX 0xFF
31#define TIMER_TOP 255 // 8 bit PWM
32
33extern backlight_config_t backlight_config;
34
35/**
36 * References
37 * Port Registers: https://www.arduino.cc/en/Reference/PortManipulation
38 * TCCR1A: https://electronics.stackexchange.com/questions/92350/what-is-the-difference-between-tccr1a-and-tccr1b
39 * Timers: http://www.avrbeginners.net/architecture/timers/timers.html
40 * 16-bit timer setup: http://sculland.com/ATmega168/Interrupts-And-Timers/16-Bit-Timer-Setup/
41 * PS2AVRGB firmware: https://github.com/showjean/ps2avrU/tree/master/firmware
42 */
43
44// @Override
45// turn LEDs on and off depending on USB caps/num/scroll lock states.
46__attribute__ ((weak))
47void led_set_user(uint8_t usb_led) {
48 if (usb_led & (1 << USB_LED_NUM_LOCK)) {
49 // turn on
50 DDRD |= NUMLOCK_PORT;
51 PORTD |= NUMLOCK_PORT;
52 } else {
53 // turn off
54 DDRD &= ~NUMLOCK_PORT;
55 PORTD &= ~NUMLOCK_PORT;
56 }
57
58 if (usb_led & (1 << USB_LED_CAPS_LOCK)) {
59 DDRD |= CAPSLOCK_PORT;
60 PORTD |= CAPSLOCK_PORT;
61 } else {
62 DDRD &= ~CAPSLOCK_PORT;
63 PORTD &= ~CAPSLOCK_PORT;
64 }
65
66 if (usb_led & (1 << USB_LED_SCROLL_LOCK)) {
67 DDRD |= SCROLLLOCK_PORT;
68 PORTD |= SCROLLLOCK_PORT;
69 } else {
70 DDRD &= ~SCROLLLOCK_PORT;
71 PORTD &= ~SCROLLLOCK_PORT;
72 }
73}
74
75#ifdef BACKLIGHT_ENABLE
76
77// sets up Timer 1 for 8-bit PWM
78void timer1PWMSetup(void) { // NOTE ONLY CALL THIS ONCE
79 // default 8 bit mode
80 TCCR1A &= ~(1 << 1); // cbi(TCCR1A,PWM11); <- set PWM11 bit to HIGH
81 TCCR1A |= (1 << 0); // sbi(TCCR1A,PWM10); <- set PWM10 bit to LOW
82
83 // clear output compare value A
84 // outb(OCR1AH, 0);
85 // outb(OCR1AL, 0);
86
87 // clear output comparator registers for B
88 OCR1BH = 0; // outb(OCR1BH, 0);
89 OCR1BL = 0; // outb(OCR1BL, 0);
90}
91
92bool is_init = false;
93void timer1Init(void) {
94 // timer1SetPrescaler(TIMER1PRESCALE)
95 // set to DIV/64
96 (TCCR1B) = ((TCCR1B) & ~TIMER_PRESCALE_MASK) | TIMER1PRESCALE;
97
98 // reset TCNT1
99 TCNT1H = 0; // outb(TCNT1H, 0);
100 TCNT1L = 0; // outb(TCNT1L, 0);
101
102 // TOIE1: Timer Overflow Interrupt Enable (Timer 1);
103 TIMSK |= _BV(TOIE1); // sbi(TIMSK, TOIE1);
104
105 is_init = true;
106}
107
108void timer1UnInit(void) {
109 // set prescaler back to NONE
110 (TCCR1B) = ((TCCR1B) & ~TIMER_PRESCALE_MASK) | 0x00; // TIMERRTC_CLK_STOP
111
112 // disable timer overflow interrupt
113 TIMSK &= ~_BV(TOIE1); // overflow bit?
114
115 setPWM(0);
116
117 is_init = false;
118}
119
120
121// handle TCNT1 overflow
122//! Interrupt handler for tcnt1 overflow interrupt
123ISR(TIMER1_OVF_vect, ISR_NOBLOCK)
124{
125 // sei();
126 // handle breathing here
127 #ifdef BACKLIGHT_BREATHING
128 if (is_breathing()) {
129 custom_breathing_handler();
130 }
131 #endif
132
133 // TODO call user defined function
134}
135
136// enable timer 1 PWM
137// timer1PWMBOn()
138void timer1PWMBEnable(void) {
139 // turn on channel B (OC1B) PWM output
140 // set OC1B as non-inverted PWM
141 TCCR1A |= _BV(COM1B1);
142 TCCR1A &= ~_BV(COM1B0);
143}
144
145// disable timer 1 PWM
146// timer1PWMBOff()
147void timer1PWMBDisable(void) {
148 TCCR1A &= ~_BV(COM1B1);
149 TCCR1A &= ~_BV(COM1B0);
150}
151
152void enableBacklight(void) {
153 DDRD |= BACKLIGHT_PORT; // set digital pin 4 as output
154 PORTD |= BACKLIGHT_PORT; // set digital pin 4 to high
155}
156
157void disableBacklight(void) {
158 // DDRD &= ~BACKLIGHT_PORT; // set digital pin 4 as input
159 PORTD &= ~BACKLIGHT_PORT; // set digital pin 4 to low
160}
161
162void startPWM(void) {
163 timer1Init();
164 timer1PWMBEnable();
165 enableBacklight();
166}
167
168void stopPWM(void) {
169 timer1UnInit();
170 disableBacklight();
171 timer1PWMBDisable();
172}
173
174void b_led_init_ports(void) {
175 /* turn backlight on/off depending on user preference */
176 #if BACKLIGHT_ON_STATE == 0
177 // DDRx register: sets the direction of Port D
178 // DDRD &= ~BACKLIGHT_PORT; // set digital pin 4 as input
179 PORTD &= ~BACKLIGHT_PORT; // set digital pin 4 to low
180 #else
181 DDRD |= BACKLIGHT_PORT; // set digital pin 4 as output
182 PORTD |= BACKLIGHT_PORT; // set digital pin 4 to high
183 #endif
184
185 timer1PWMSetup();
186 startPWM();
187
188 #ifdef BACKLIGHT_BREATHING
189 breathing_enable();
190 #endif
191}
192
193void b_led_set(uint8_t level) {
194 if (level > BACKLIGHT_LEVELS) {
195 level = BACKLIGHT_LEVELS;
196 }
197
198 setPWM((int)(TIMER_TOP * (float) level / BACKLIGHT_LEVELS));
199}
200
201// called every matrix scan
202void b_led_task(void) {
203 // do nothing for now
204}
205
206void setPWM(uint16_t xValue) {
207 if (xValue > TIMER_TOP) {
208 xValue = TIMER_TOP;
209 }
210 OCR1B = xValue; // timer1PWMBSet(xValue);
211}
212
213#endif // BACKLIGHT_ENABLE
diff --git a/keyboards/jj4x4/backlight_custom.h b/keyboards/jj4x4/backlight_custom.h
new file mode 100644
index 000000000..7210be840
--- /dev/null
+++ b/keyboards/jj4x4/backlight_custom.h
@@ -0,0 +1,15 @@
1/**
2 * Backlighting code for PS2AVRGB boards (ATMEGA32A)
3 * Kenneth A. (github.com/krusli | krusli.me)
4 */
5
6#ifndef BACKLIGHT_CUSTOM_H
7#define BACKLIGHT_CUSTOM_H
8
9#include <avr/pgmspace.h>
10void b_led_init_ports(void);
11void b_led_set(uint8_t level);
12void b_led_task(void);
13void setPWM(uint16_t xValue);
14
15#endif // BACKLIGHT_CUSTOM_H
diff --git a/keyboards/jj4x4/breathing_custom.h b/keyboards/jj4x4/breathing_custom.h
new file mode 100644
index 000000000..71416b1b4
--- /dev/null
+++ b/keyboards/jj4x4/breathing_custom.h
@@ -0,0 +1,140 @@
1/**
2 * Breathing effect code for PS2AVRGB boards (ATMEGA32A)
3 * Works in conjunction with `backlight.c`.
4 *
5 * Code adapted from `quantum.c` to register with the existing TIMER1 overflow
6 * handler in `backlight.c` instead of setting up its own timer.
7 * Kenneth A. (github.com/krusli | krusli.me)
8 */
9
10#ifdef BACKLIGHT_ENABLE
11#ifdef BACKLIGHT_BREATHING
12
13#include "backlight_custom.h"
14
15#ifndef BREATHING_PERIOD
16#define BREATHING_PERIOD 6
17#endif
18
19#define breathing_min() do {breathing_counter = 0;} while (0)
20#define breathing_max() do {breathing_counter = breathing_period * 244 / 2;} while (0)
21
22// TODO make this share code with quantum.c
23
24#define BREATHING_NO_HALT 0
25#define BREATHING_HALT_OFF 1
26#define BREATHING_HALT_ON 2
27#define BREATHING_STEPS 128
28
29static uint8_t breathing_period = BREATHING_PERIOD;
30static uint8_t breathing_halt = BREATHING_NO_HALT;
31static uint16_t breathing_counter = 0;
32
33static bool breathing = false;
34
35bool is_breathing(void) {
36 return breathing;
37}
38
39// See http://jared.geek.nz/2013/feb/linear-led-pwm
40static uint16_t cie_lightness(uint16_t v) {
41 if (v <= 5243) // if below 8% of max
42 return v / 9; // same as dividing by 900%
43 else {
44 uint32_t y = (((uint32_t) v + 10486) << 8) / (10486 + 0xFFFFUL); // add 16% of max and compare
45 // to get a useful result with integer division, we shift left in the expression above
46 // and revert what we've done again after squaring.
47 y = y * y * y >> 8;
48 if (y > 0xFFFFUL) // prevent overflow
49 return 0xFFFFU;
50 else
51 return (uint16_t) y;
52 }
53}
54
55void breathing_enable(void) {
56 breathing = true;
57 breathing_counter = 0;
58 breathing_halt = BREATHING_NO_HALT;
59 // interrupt already registered
60}
61
62void breathing_pulse(void) {
63 if (get_backlight_level() == 0)
64 breathing_min();
65 else
66 breathing_max();
67 breathing_halt = BREATHING_HALT_ON;
68 // breathing_interrupt_enable();
69 breathing = true;
70}
71
72void breathing_disable(void) {
73 breathing = false;
74 // backlight_set(get_backlight_level());
75 b_led_set(get_backlight_level()); // custom implementation of backlight_set()
76}
77
78void breathing_self_disable(void)
79{
80 if (get_backlight_level() == 0)
81 breathing_halt = BREATHING_HALT_OFF;
82 else
83 breathing_halt = BREATHING_HALT_ON;
84}
85
86void breathing_toggle(void) {
87 if (is_breathing())
88 breathing_disable();
89 else
90 breathing_enable();
91}
92
93void breathing_period_set(uint8_t value)
94{
95 if (!value)
96 value = 1;
97 breathing_period = value;
98}
99
100void breathing_period_default(void) {
101 breathing_period_set(BREATHING_PERIOD);
102}
103
104void breathing_period_inc(void)
105{
106 breathing_period_set(breathing_period+1);
107}
108
109void breathing_period_dec(void)
110{
111 breathing_period_set(breathing_period-1);
112}
113
114/* To generate breathing curve in python:
115 * from math import sin, pi; [int(sin(x/128.0*pi)**4*255) for x in range(128)]
116 */
117static const uint8_t breathing_table[BREATHING_STEPS] PROGMEM = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 2, 3, 4, 5, 6, 8, 10, 12, 15, 17, 20, 24, 28, 32, 36, 41, 46, 51, 57, 63, 70, 76, 83, 91, 98, 106, 113, 121, 129, 138, 146, 154, 162, 170, 178, 185, 193, 200, 207, 213, 220, 225, 231, 235, 240, 244, 247, 250, 252, 253, 254, 255, 254, 253, 252, 250, 247, 244, 240, 235, 231, 225, 220, 213, 207, 200, 193, 185, 178, 170, 162, 154, 146, 138, 129, 121, 113, 106, 98, 91, 83, 76, 70, 63, 57, 51, 46, 41, 36, 32, 28, 24, 20, 17, 15, 12, 10, 8, 6, 5, 4, 3, 2, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
118
119// Use this before the cie_lightness function.
120static inline uint16_t scale_backlight(uint16_t v) {
121 return v / BACKLIGHT_LEVELS * get_backlight_level();
122}
123
124void custom_breathing_handler(void) {
125 uint16_t interval = (uint16_t) breathing_period * 244 / BREATHING_STEPS;
126 // resetting after one period to prevent ugly reset at overflow.
127 breathing_counter = (breathing_counter + 1) % (breathing_period * 244);
128 uint8_t index = breathing_counter / interval % BREATHING_STEPS;
129
130 if (((breathing_halt == BREATHING_HALT_ON) && (index == BREATHING_STEPS / 2)) ||
131 ((breathing_halt == BREATHING_HALT_OFF) && (index == BREATHING_STEPS - 1)))
132 {
133 // breathing_interrupt_disable();
134 }
135
136 setPWM(cie_lightness(scale_backlight((uint16_t) pgm_read_byte(&breathing_table[index]) * 0x0101U)));
137}
138
139#endif // BACKLIGHT_BREATHING
140#endif // BACKLIGHT_ENABLE
diff --git a/keyboards/jj4x4/config.h b/keyboards/jj4x4/config.h
new file mode 100644
index 000000000..d48df5f1d
--- /dev/null
+++ b/keyboards/jj4x4/config.h
@@ -0,0 +1,53 @@
1/*
2Copyright 2017 Luiz Ribeiro <luizribeiro@gmail.com>
3
4This program is free software: you can redistribute it and/or modify
5it under the terms of the GNU General Public License as published by
6the Free Software Foundation, either version 2 of the License, or
7(at your option) any later version.
8
9This program is distributed in the hope that it will be useful,
10but WITHOUT ANY WARRANTY; without even the implied warranty of
11MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12GNU General Public License for more details.
13
14You should have received a copy of the GNU General Public License
15along with this program. If not, see <http://www.gnu.org/licenses/>.
16*/
17
18#include "config_common.h"
19
20#ifndef CONFIG_H
21#define CONFIG_H
22
23#define VENDOR_ID 0x20A0
24#define PRODUCT_ID 0x422D
25// TODO: share these strings with usbconfig.h
26// Edit usbconfig.h to change these.
27#define MANUFACTURER Kprepublic
28#define PRODUCT jj4x4
29
30/* matrix size */
31#define MATRIX_ROWS 4
32#define MATRIX_COLS 4
33#define MATRIX_ROW_PINS { B5, B0, B3, B4 }
34#define MATRIX_COL_PINS { A1, A0, A2, A3 }
35
36/* COL2ROW or ROW2COL */
37#define DIODE_DIRECTION COL2ROW
38
39#define BACKLIGHT_LEVELS 12
40// #define BACKLIGHT_BREATHING // works, but BL_TOGG might not work
41
42#define TAPPING_TOGGLE 3
43
44#define NO_UART 1
45
46/* RGB underglow */
47// The RGB_DI_PIN value seems to be shared between all PS2AVRGB boards.
48// The same pin is used on the JJ40, at least.
49#define RGBLED_NUM 5
50#define RGB_DI_PIN E2 // NOTE: for PS2AVRGB boards, underglow commands are sent via I2C to 0xB0.
51#define RGBLIGHT_ANIMATIONS
52
53#endif
diff --git a/keyboards/jj4x4/i2c.c b/keyboards/jj4x4/i2c.c
new file mode 100644
index 000000000..c27f3e3d1
--- /dev/null
+++ b/keyboards/jj4x4/i2c.c
@@ -0,0 +1,104 @@
1/*
2Copyright 2016 Luiz Ribeiro <luizribeiro@gmail.com>
3
4This program is free software: you can redistribute it and/or modify
5it under the terms of the GNU General Public License as published by
6the Free Software Foundation, either version 2 of the License, or
7(at your option) any later version.
8
9This program is distributed in the hope that it will be useful,
10but WITHOUT ANY WARRANTY; without even the implied warranty of
11MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12GNU General Public License for more details.
13
14You should have received a copy of the GNU General Public License
15along with this program. If not, see <http://www.gnu.org/licenses/>.
16*/
17
18#include <avr/io.h>
19#include <util/twi.h>
20
21#include "i2c.h"
22
23void i2c_set_bitrate(uint16_t bitrate_khz) {
24 uint8_t bitrate_div = ((F_CPU / 1000l) / bitrate_khz);
25 if (bitrate_div >= 16) {
26 bitrate_div = (bitrate_div - 16) / 2;
27 }
28 TWBR = bitrate_div;
29}
30
31void i2c_init(void) {
32 // set pull-up resistors on I2C bus pins
33 PORTC |= 0b11;
34
35 i2c_set_bitrate(400);
36
37 // enable TWI (two-wire interface)
38 TWCR |= (1 << TWEN);
39
40 // enable TWI interrupt and slave address ACK
41 TWCR |= (1 << TWIE);
42 TWCR |= (1 << TWEA);
43}
44
45uint8_t i2c_start(uint8_t address) {
46 // reset TWI control register
47 TWCR = 0;
48
49 // begin transmission and wait for it to end
50 TWCR = (1<<TWINT) | (1<<TWSTA) | (1<<TWEN);
51 while (!(TWCR & (1<<TWINT)));
52
53 // check if the start condition was successfully transmitted
54 if ((TWSR & 0xF8) != TW_START) {
55 return 1;
56 }
57
58 // transmit address and wait
59 TWDR = address;
60 TWCR = (1<<TWINT) | (1<<TWEN);
61 while (!(TWCR & (1<<TWINT)));
62
63 // check if the device has acknowledged the READ / WRITE mode
64 uint8_t twst = TW_STATUS & 0xF8;
65 if ((twst != TW_MT_SLA_ACK) && (twst != TW_MR_SLA_ACK)) {
66 return 1;
67 }
68
69 return 0;
70}
71
72void i2c_stop(void) {
73 TWCR = (1<<TWINT) | (1<<TWEN) | (1<<TWSTO);
74}
75
76uint8_t i2c_write(uint8_t data) {
77 TWDR = data;
78
79 // transmit data and wait
80 TWCR = (1<<TWINT) | (1<<TWEN);
81 while (!(TWCR & (1<<TWINT)));
82
83 if ((TWSR & 0xF8) != TW_MT_DATA_ACK) {
84 return 1;
85 }
86
87 return 0;
88}
89
90uint8_t i2c_send(uint8_t address, uint8_t *data, uint16_t length) {
91 if (i2c_start(address)) {
92 return 1;
93 }
94
95 for (uint16_t i = 0; i < length; i++) {
96 if (i2c_write(data[i])) {
97 return 1;
98 }
99 }
100
101 i2c_stop();
102
103 return 0;
104}
diff --git a/keyboards/jj4x4/i2c.h b/keyboards/jj4x4/i2c.h
new file mode 100644
index 000000000..27c9d3d05
--- /dev/null
+++ b/keyboards/jj4x4/i2c.h
@@ -0,0 +1,25 @@
1/*
2Copyright 2016 Luiz Ribeiro <luizribeiro@gmail.com>
3
4This program is free software: you can redistribute it and/or modify
5it under the terms of the GNU General Public License as published by
6the Free Software Foundation, either version 2 of the License, or
7(at your option) any later version.
8
9This program is distributed in the hope that it will be useful,
10but WITHOUT ANY WARRANTY; without even the implied warranty of
11MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12GNU General Public License for more details.
13
14You should have received a copy of the GNU General Public License
15along with this program. If not, see <http://www.gnu.org/licenses/>.
16*/
17
18#ifndef __I2C_H__
19#define __I2C_H__
20
21void i2c_init(void);
22void i2c_set_bitrate(uint16_t bitrate_khz);
23uint8_t i2c_send(uint8_t address, uint8_t *data, uint16_t length);
24
25#endif
diff --git a/keyboards/jj4x4/info.json b/keyboards/jj4x4/info.json
new file mode 100644
index 000000000..c6416ce16
--- /dev/null
+++ b/keyboards/jj4x4/info.json
@@ -0,0 +1,23 @@
1{
2 "keyboard_name": "jj40",
3 "url": "",
4 "maintainer": "qmk",
5 "width": 12,
6 "height": 4,
7 "layouts": {
8 "LAYOUT_planck_mit": {
9 "key_count": 47,
10 "layout": [{"x":0, "y":0}, {"x":1, "y":0}, {"x":2, "y":0}, {"x":3, "y":0}, {"x":4, "y":0}, {"x":5, "y":0}, {"x":6, "y":0}, {"x":7, "y":0}, {"x":8, "y":0}, {"x":9, "y":0}, {"x":10, "y":0}, {"x":11, "y":0}, {"x":0, "y":1}, {"x":1, "y":1}, {"x":2, "y":1}, {"x":3, "y":1}, {"x":4, "y":1}, {"x":5, "y":1}, {"x":6, "y":1}, {"x":7, "y":1}, {"x":8, "y":1}, {"x":9, "y":1}, {"x":10, "y":1}, {"x":11, "y":1}, {"x":0, "y":2}, {"x":1, "y":2}, {"x":2, "y":2}, {"x":3, "y":2}, {"x":4, "y":2}, {"x":5, "y":2}, {"x":6, "y":2}, {"x":7, "y":2}, {"x":8, "y":2}, {"x":9, "y":2}, {"x":10, "y":2}, {"x":11, "y":2}, {"x":0, "y":3}, {"x":1, "y":3}, {"x":2, "y":3}, {"x":3, "y":3}, {"x":4, "y":3}, {"x":5, "y":3, "w":2}, {"x":7, "y":3}, {"x":8, "y":3}, {"x":9, "y":3}, {"x":10, "y":3}, {"x":11, "y":3}]
11 },
12
13 "LAYOUT_ortho_4x12": {
14 "key_count": 48,
15 "layout": [{"x":0, "y":0}, {"x":1, "y":0}, {"x":2, "y":0}, {"x":3, "y":0}, {"x":4, "y":0}, {"x":5, "y":0}, {"x":6, "y":0}, {"x":7, "y":0}, {"x":8, "y":0}, {"x":9, "y":0}, {"x":10, "y":0}, {"x":11, "y":0}, {"x":0, "y":1}, {"x":1, "y":1}, {"x":2, "y":1}, {"x":3, "y":1}, {"x":4, "y":1}, {"x":5, "y":1}, {"x":6, "y":1}, {"x":7, "y":1}, {"x":8, "y":1}, {"x":9, "y":1}, {"x":10, "y":1}, {"x":11, "y":1}, {"x":0, "y":2}, {"x":1, "y":2}, {"x":2, "y":2}, {"x":3, "y":2}, {"x":4, "y":2}, {"x":5, "y":2}, {"x":6, "y":2}, {"x":7, "y":2}, {"x":8, "y":2}, {"x":9, "y":2}, {"x":10, "y":2}, {"x":11, "y":2}, {"x":0, "y":3}, {"x":1, "y":3}, {"x":2, "y":3}, {"x":3, "y":3}, {"x":4, "y":3}, {"x":5, "y":3}, {"x":6, "y":3}, {"x":7, "y":3}, {"x":8, "y":3}, {"x":9, "y":3}, {"x":10, "y":3}, {"x":11, "y":3}]
16 },
17
18 "LAYOUT_planck_1x2uR": {
19 "key_count": 47,
20 "layout": [{"x":0, "y":0}, {"x":1, "y":0}, {"x":2, "y":0}, {"x":3, "y":0}, {"x":4, "y":0}, {"x":5, "y":0}, {"x":6, "y":0}, {"x":7, "y":0}, {"x":8, "y":0}, {"x":9, "y":0}, {"x":10, "y":0}, {"x":11, "y":0}, {"x":0, "y":1}, {"x":1, "y":1}, {"x":2, "y":1}, {"x":3, "y":1}, {"x":4, "y":1}, {"x":5, "y":1}, {"x":6, "y":1}, {"x":7, "y":1}, {"x":8, "y":1}, {"x":9, "y":1}, {"x":10, "y":1}, {"x":11, "y":1}, {"x":0, "y":2}, {"x":1, "y":2}, {"x":2, "y":2}, {"x":3, "y":2}, {"x":4, "y":2}, {"x":5, "y":2}, {"x":6, "y":2}, {"x":7, "y":2}, {"x":8, "y":2}, {"x":9, "y":2}, {"x":10, "y":2}, {"x":11, "y":2}, {"x":0, "y":3}, {"x":1, "y":3}, {"x":2, "y":3}, {"x":3, "y":3}, {"x":4, "y":3}, {"x":5, "y":3}, {"x":6, "y":3, "w":2}, {"x":8, "y":3}, {"x":9, "y":3}, {"x":10, "y":3}, {"x":11, "y":3}]
21 }
22 }
23}
diff --git a/keyboards/jj4x4/jj4x4.c b/keyboards/jj4x4/jj4x4.c
new file mode 100644
index 000000000..6fbff7afb
--- /dev/null
+++ b/keyboards/jj4x4/jj4x4.c
@@ -0,0 +1,97 @@
1/*
2Copyright 2017 Luiz Ribeiro <luizribeiro@gmail.com>
3Modified 2018 Kenneth A. <github.com/krusli>
4
5This program is free software: you can redistribute it and/or modify
6it under the terms of the GNU General Public License as published by
7the Free Software Foundation, either version 2 of the License, or
8(at your option) any later version.
9
10This program is distributed in the hope that it will be useful,
11but WITHOUT ANY WARRANTY; without even the implied warranty of
12MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13GNU General Public License for more details.
14
15You should have received a copy of the GNU General Public License
16along with this program. If not, see <http://www.gnu.org/licenses/>.
17*/
18
19#include "jj4x4.h"
20
21#include <avr/pgmspace.h>
22
23#include "action_layer.h"
24#include "quantum.h"
25
26#include "i2c.h"
27
28#include "backlight.h"
29#include "backlight_custom.h"
30
31// for keyboard subdirectory level init functions
32// @Override
33void matrix_init_kb(void) {
34 // call user level keymaps, if any
35 matrix_init_user();
36}
37
38#ifdef BACKLIGHT_ENABLE
39/// Overrides functions in `quantum.c`
40void backlight_init_ports(void) {
41 b_led_init_ports();
42}
43
44void backlight_task(void) {
45 b_led_task();
46}
47
48void backlight_set(uint8_t level) {
49 b_led_set(level);
50}
51#endif
52
53#ifdef RGBLIGHT_ENABLE
54extern rgblight_config_t rgblight_config;
55
56// custom RGB driver
57void rgblight_set(void) {
58 if (!rgblight_config.enable) {
59 for (uint8_t i=0; i<RGBLED_NUM; i++) {
60 led[i].r = 0;
61 led[i].g = 0;
62 led[i].b = 0;
63 }
64 }
65
66 i2c_init();
67 i2c_send(0xb0, (uint8_t*)led, 3 * RGBLED_NUM);
68}
69
70bool rgb_init = false;
71
72void matrix_scan_kb(void) {
73 // if LEDs were previously on before poweroff, turn them back on
74 if (rgb_init == false && rgblight_config.enable) {
75 i2c_init();
76 i2c_send(0xb0, (uint8_t*)led, 3 * RGBLED_NUM);
77 rgb_init = true;
78 }
79
80 rgblight_task();
81#else
82void matrix_scan_kb(void) {
83#endif
84 matrix_scan_user();
85 /* Nothing else for now. */
86}
87
88__attribute__((weak)) // overridable
89void matrix_init_user(void) {
90
91}
92
93
94__attribute__((weak)) // overridable
95void matrix_scan_user(void) {
96
97}
diff --git a/keyboards/jj4x4/jj4x4.h b/keyboards/jj4x4/jj4x4.h
new file mode 100644
index 000000000..03cf5d8a6
--- /dev/null
+++ b/keyboards/jj4x4/jj4x4.h
@@ -0,0 +1,46 @@
1/*
2Copyright 2017 Luiz Ribeiro <luizribeiro@gmail.com>
3
4This program is free software: you can redistribute it and/or modify
5it under the terms of the GNU General Public License as published by
6the Free Software Foundation, either version 2 of the License, or
7(at your option) any later version.
8
9This program is distributed in the hope that it will be useful,
10but WITHOUT ANY WARRANTY; without even the implied warranty of
11MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12GNU General Public License for more details.
13
14You should have received a copy of the GNU General Public License
15along with this program. If not, see <http://www.gnu.org/licenses/>.
16*/
17
18#ifndef KEYMAP_COMMON_H
19#define KEYMAP_COMMON_H
20
21#include "quantum.h"
22#include "quantum_keycodes.h"
23#include "keycode.h"
24#include "action.h"
25
26void matrix_init_user(void); // TODO port this to other PS2AVRGB boards
27
28#define XXX KC_NO
29
30
31#define LAYOUT( \
32 K01, K02, K03, K04, \
33 K11, K12, K13, K14, \
34 K21, K22, K23, K24, \
35 K31, K32, K33, K34 \
36) \
37{ \
38 { K01, K02, K03, K04 }, \
39 { K11, K12, K13, K14 }, \
40 { K21, K22, K23, K24 }, \
41 { K31, K32, K33, K34 } \
42}
43
44
45
46#endif
diff --git a/keyboards/jj4x4/keymaps/default/keymap.c b/keyboards/jj4x4/keymaps/default/keymap.c
new file mode 100644
index 000000000..d4130e809
--- /dev/null
+++ b/keyboards/jj4x4/keymaps/default/keymap.c
@@ -0,0 +1,34 @@
1#include QMK_KEYBOARD_H
2
3
4#define _QWERTY 0
5
6
7enum custom_keycodes {
8 QWERTY = SAFE_RANGE,
9};
10
11const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
12/* Qwerty
13 * ,---------------------------.
14 * | A | B | C | D |
15 * +------+------+------+------|
16 * | E | F | G | H |
17 * +------+------+------+------|
18 * | I | J | K | L |
19 * +------+------+------+------|
20 * | M | N | O | P |
21 * ----------------------------'
22 */
23[_QWERTY] = LAYOUT( \
24 KC_A, KC_B, KC_C, KC_D, \
25 KC_E, KC_F, KC_G, KC_H, \
26 KC_I, KC_J, KC_K, KC_L, \
27 KC_M, KC_N, KC_O, KC_P \
28),
29};
30
31// Loop
32void matrix_scan_user(void) {
33 // Empty
34};
diff --git a/keyboards/jj4x4/rules.mk b/keyboards/jj4x4/rules.mk
new file mode 100644
index 000000000..3e222c620
--- /dev/null
+++ b/keyboards/jj4x4/rules.mk
@@ -0,0 +1,58 @@
1# Copyright 2017 Luiz Ribeiro <luizribeiro@gmail.com>
2#
3# This program is free software: you can redistribute it and/or modify
4# it under the terms of the GNU General Public License as published by
5# the Free Software Foundation, either version 2 of the License, or
6# (at your option) any later version.
7#
8# This program is distributed in the hope that it will be useful,
9# but WITHOUT ANY WARRANTY; without even the implied warranty of
10# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11# GNU General Public License for more details.
12#
13# You should have received a copy of the GNU General Public License
14# along with this program. If not, see <http://www.gnu.org/licenses/>.
15
16# MCU name
17MCU = atmega32a
18PROTOCOL = VUSB
19
20# unsupported features for now
21NO_UART = yes
22NO_SUSPEND_POWER_DOWN = yes
23
24# processor frequency
25F_CPU = 12000000
26
27# Bootloader
28# This definition is optional, and if your keyboard supports multiple bootloaders of
29# different sizes, comment this out, and the correct address will be loaded
30# automatically (+60). See bootloader.mk for all options.
31BOOTLOADER = bootloadHID
32
33# build options
34BOOTMAGIC_ENABLE = no
35MOUSEKEY_ENABLE = no
36EXTRAKEY_ENABLE = yes
37CONSOLE_ENABLE = no
38COMMAND_ENABLE = yes
39
40BACKLIGHT_ENABLE = yes
41BACKLIGHT_CUSTOM_DRIVER = yes
42
43RGBLIGHT_ENABLE = yes
44RGBLIGHT_CUSTOM_DRIVER = yes
45
46KEY_LOCK_ENABLE = yes
47
48# Do not enable SLEEP_LED_ENABLE. it uses the same timer as BACKLIGHT_ENABLE
49SLEEP_LED_ENABLE = no # Breathing sleep LED during USB suspend
50
51OPT_DEFS = -DDEBUG_LEVEL=0
52
53SRC = i2c.c backlight.c
54
55# programming options
56PROGRAM_CMD = ./util/atmega32a_program.py $(TARGET).hex
57
58
diff --git a/keyboards/jj4x4/tools/README.md b/keyboards/jj4x4/tools/README.md
new file mode 100644
index 000000000..081984ed4
--- /dev/null
+++ b/keyboards/jj4x4/tools/README.md
@@ -0,0 +1,16 @@
1# JJ40 Tools
2
3## usb_detach.c
4
5When trying to flash on Linux, you may encounter a "Resource Unavailable" error. This means that Linux's HID driver has taken exclusive control of the keyboard, and the program script can't flash it.
6This program can force Linux to give up a device, so that the programming script can reset it.
7
8### To compile:
9```
10gcc usb_detach.c -o usb_detach
11```
12
13### To run:
141. Use `lsusb` to discover the Bus and Device numbers for your keyboard.
152. Run the program: `sudo ./usb_detach /dev/bus/usb/<BUS>/<DEVICE>`.
163. Build and program the firmware as normal.
diff --git a/keyboards/jj4x4/tools/usb_detach.c b/keyboards/jj4x4/tools/usb_detach.c
new file mode 100644
index 000000000..c21e47a7a
--- /dev/null
+++ b/keyboards/jj4x4/tools/usb_detach.c
@@ -0,0 +1,33 @@
1/* Found at https://www.linuxquestions.org/questions/linux-hardware-18/how-to-unclaim-usb-device-558138/#post3406986 */
2#include <stdio.h>
3#include <sys/types.h>
4#include <sys/stat.h>
5#include <fcntl.h>
6#include <linux/ioctl.h>
7#include <linux/usbdevice_fs.h>
8
9int main(int argc, char**argv)
10{
11 struct usbdevfs_ioctl command;
12 int ret;
13 int fd;
14 int i;
15 if (argc>1) {
16 fd = open(argv[1],O_RDWR);
17 if (fd<1){
18 perror("unable to open file");
19 return 1;
20 }
21 for (i=0;i<255;i++){ // hack: should fetch how many interface there is.
22 command.ifno = i;
23 command.ioctl_code = USBDEVFS_DISCONNECT;
24 command.data = NULL;
25 ret = ioctl(fd, USBDEVFS_IOCTL, &command);
26 if(ret!=-1)
27 printf("un claimed interface %d %d\n",i,ret);
28 }
29 } else {
30 printf ("usage: %s /dev/bus/usb/BUS/DEVICE\n",argv[0]);
31 printf("Release all interfaces of this usb device for usage in virtualisation\n");
32 }
33} \ No newline at end of file
diff --git a/keyboards/jj4x4/usbconfig.h b/keyboards/jj4x4/usbconfig.h
new file mode 100644
index 000000000..ad97e7f0a
--- /dev/null
+++ b/keyboards/jj4x4/usbconfig.h
@@ -0,0 +1,397 @@
1/* Name: usbconfig.h
2 * Project: V-USB, virtual USB port for Atmel's(r) AVR(r) microcontrollers
3 * Author: Christian Starkjohann
4 * Creation Date: 2005-04-01
5 * Tabsize: 4
6 * Copyright: (c) 2005 by OBJECTIVE DEVELOPMENT Software GmbH
7 * License: GNU GPL v2 (see License.txt), GNU GPL v3 or proprietary (CommercialLicense.txt)
8 * This Revision: $Id: usbconfig-prototype.h 785 2010-05-30 17:57:07Z cs $
9 */
10
11#ifndef __usbconfig_h_included__
12#define __usbconfig_h_included__
13
14#include "config.h"
15
16/*
17General Description:
18This file is an example configuration (with inline documentation) for the USB
19driver. It configures V-USB for USB D+ connected to Port D bit 2 (which is
20also hardware interrupt 0 on many devices) and USB D- to Port D bit 4. You may
21wire the lines to any other port, as long as D+ is also wired to INT0 (or any
22other hardware interrupt, as long as it is the highest level interrupt, see
23section at the end of this file).
24*/
25
26/* ---------------------------- Hardware Config ---------------------------- */
27
28#define USB_CFG_IOPORTNAME D
29/* This is the port where the USB bus is connected. When you configure it to
30 * "B", the registers PORTB, PINB and DDRB will be used.
31 */
32#define USB_CFG_DMINUS_BIT 3
33/* This is the bit number in USB_CFG_IOPORT where the USB D- line is connected.
34 * This may be any bit in the port.
35 */
36#define USB_CFG_DPLUS_BIT 2
37/* This is the bit number in USB_CFG_IOPORT where the USB D+ line is connected.
38 * This may be any bit in the port. Please note that D+ must also be connected
39 * to interrupt pin INT0! [You can also use other interrupts, see section
40 * "Optional MCU Description" below, or you can connect D- to the interrupt, as
41 * it is required if you use the USB_COUNT_SOF feature. If you use D- for the
42 * interrupt, the USB interrupt will also be triggered at Start-Of-Frame
43 * markers every millisecond.]
44 */
45#define USB_CFG_CLOCK_KHZ (F_CPU/1000)
46/* Clock rate of the AVR in kHz. Legal values are 12000, 12800, 15000, 16000,
47 * 16500, 18000 and 20000. The 12.8 MHz and 16.5 MHz versions of the code
48 * require no crystal, they tolerate +/- 1% deviation from the nominal
49 * frequency. All other rates require a precision of 2000 ppm and thus a
50 * crystal!
51 * Since F_CPU should be defined to your actual clock rate anyway, you should
52 * not need to modify this setting.
53 */
54#define USB_CFG_CHECK_CRC 0
55/* Define this to 1 if you want that the driver checks integrity of incoming
56 * data packets (CRC checks). CRC checks cost quite a bit of code size and are
57 * currently only available for 18 MHz crystal clock. You must choose
58 * USB_CFG_CLOCK_KHZ = 18000 if you enable this option.
59 */
60
61/* ----------------------- Optional Hardware Config ------------------------ */
62
63/* #define USB_CFG_PULLUP_IOPORTNAME D */
64/* If you connect the 1.5k pullup resistor from D- to a port pin instead of
65 * V+, you can connect and disconnect the device from firmware by calling
66 * the macros usbDeviceConnect() and usbDeviceDisconnect() (see usbdrv.h).
67 * This constant defines the port on which the pullup resistor is connected.
68 */
69/* #define USB_CFG_PULLUP_BIT 4 */
70/* This constant defines the bit number in USB_CFG_PULLUP_IOPORT (defined
71 * above) where the 1.5k pullup resistor is connected. See description
72 * above for details.
73 */
74
75/* --------------------------- Functional Range ---------------------------- */
76
77#define USB_CFG_HAVE_INTRIN_ENDPOINT 1
78/* Define this to 1 if you want to compile a version with two endpoints: The
79 * default control endpoint 0 and an interrupt-in endpoint (any other endpoint
80 * number).
81 */
82#define USB_CFG_HAVE_INTRIN_ENDPOINT3 1
83/* Define this to 1 if you want to compile a version with three endpoints: The
84 * default control endpoint 0, an interrupt-in endpoint 3 (or the number
85 * configured below) and a catch-all default interrupt-in endpoint as above.
86 * You must also define USB_CFG_HAVE_INTRIN_ENDPOINT to 1 for this feature.
87 */
88#define USB_CFG_EP3_NUMBER 3
89/* If the so-called endpoint 3 is used, it can now be configured to any other
90 * endpoint number (except 0) with this macro. Default if undefined is 3.
91 */
92/* #define USB_INITIAL_DATATOKEN USBPID_DATA1 */
93/* The above macro defines the startup condition for data toggling on the
94 * interrupt/bulk endpoints 1 and 3. Defaults to USBPID_DATA1.
95 * Since the token is toggled BEFORE sending any data, the first packet is
96 * sent with the oposite value of this configuration!
97 */
98#define USB_CFG_IMPLEMENT_HALT 0
99/* Define this to 1 if you also want to implement the ENDPOINT_HALT feature
100 * for endpoint 1 (interrupt endpoint). Although you may not need this feature,
101 * it is required by the standard. We have made it a config option because it
102 * bloats the code considerably.
103 */
104#define USB_CFG_SUPPRESS_INTR_CODE 0
105/* Define this to 1 if you want to declare interrupt-in endpoints, but don't
106 * want to send any data over them. If this macro is defined to 1, functions
107 * usbSetInterrupt() and usbSetInterrupt3() are omitted. This is useful if
108 * you need the interrupt-in endpoints in order to comply to an interface
109 * (e.g. HID), but never want to send any data. This option saves a couple
110 * of bytes in flash memory and the transmit buffers in RAM.
111 */
112#define USB_CFG_INTR_POLL_INTERVAL 1
113/* If you compile a version with endpoint 1 (interrupt-in), this is the poll
114 * interval. The value is in milliseconds and must not be less than 10 ms for
115 * low speed devices.
116 */
117#define USB_CFG_IS_SELF_POWERED 0
118/* Define this to 1 if the device has its own power supply. Set it to 0 if the
119 * device is powered from the USB bus.
120 */
121// max power draw with maxed white underglow measured at 120 mA (peaks)
122#define USB_CFG_MAX_BUS_POWER 100
123/* Set this variable to the maximum USB bus power consumption of your device.
124 * The value is in milliamperes. [It will be divided by two since USB
125 * communicates power requirements in units of 2 mA.]
126 */
127#define USB_CFG_IMPLEMENT_FN_WRITE 1
128/* Set this to 1 if you want usbFunctionWrite() to be called for control-out
129 * transfers. Set it to 0 if you don't need it and want to save a couple of
130 * bytes.
131 */
132#define USB_CFG_IMPLEMENT_FN_READ 0
133/* Set this to 1 if you need to send control replies which are generated
134 * "on the fly" when usbFunctionRead() is called. If you only want to send
135 * data from a static buffer, set it to 0 and return the data from
136 * usbFunctionSetup(). This saves a couple of bytes.
137 */
138#define USB_CFG_IMPLEMENT_FN_WRITEOUT 0
139/* Define this to 1 if you want to use interrupt-out (or bulk out) endpoints.
140 * You must implement the function usbFunctionWriteOut() which receives all
141 * interrupt/bulk data sent to any endpoint other than 0. The endpoint number
142 * can be found in 'usbRxToken'.
143 */
144#define USB_CFG_HAVE_FLOWCONTROL 0
145/* Define this to 1 if you want flowcontrol over USB data. See the definition
146 * of the macros usbDisableAllRequests() and usbEnableAllRequests() in
147 * usbdrv.h.
148 */
149#define USB_CFG_DRIVER_FLASH_PAGE 0
150/* If the device has more than 64 kBytes of flash, define this to the 64 k page
151 * where the driver's constants (descriptors) are located. Or in other words:
152 * Define this to 1 for boot loaders on the ATMega128.
153 */
154#define USB_CFG_LONG_TRANSFERS 0
155/* Define this to 1 if you want to send/receive blocks of more than 254 bytes
156 * in a single control-in or control-out transfer. Note that the capability
157 * for long transfers increases the driver size.
158 */
159/* #define USB_RX_USER_HOOK(data, len) if(usbRxToken == (uchar)USBPID_SETUP) blinkLED(); */
160/* This macro is a hook if you want to do unconventional things. If it is
161 * defined, it's inserted at the beginning of received message processing.
162 * If you eat the received message and don't want default processing to
163 * proceed, do a return after doing your things. One possible application
164 * (besides debugging) is to flash a status LED on each packet.
165 */
166/* #define USB_RESET_HOOK(resetStarts) if(!resetStarts){hadUsbReset();} */
167/* This macro is a hook if you need to know when an USB RESET occurs. It has
168 * one parameter which distinguishes between the start of RESET state and its
169 * end.
170 */
171/* #define USB_SET_ADDRESS_HOOK() hadAddressAssigned(); */
172/* This macro (if defined) is executed when a USB SET_ADDRESS request was
173 * received.
174 */
175#define USB_COUNT_SOF 1
176/* define this macro to 1 if you need the global variable "usbSofCount" which
177 * counts SOF packets. This feature requires that the hardware interrupt is
178 * connected to D- instead of D+.
179 */
180/* #ifdef __ASSEMBLER__
181 * macro myAssemblerMacro
182 * in YL, TCNT0
183 * sts timer0Snapshot, YL
184 * endm
185 * #endif
186 * #define USB_SOF_HOOK myAssemblerMacro
187 * This macro (if defined) is executed in the assembler module when a
188 * Start Of Frame condition is detected. It is recommended to define it to
189 * the name of an assembler macro which is defined here as well so that more
190 * than one assembler instruction can be used. The macro may use the register
191 * YL and modify SREG. If it lasts longer than a couple of cycles, USB messages
192 * immediately after an SOF pulse may be lost and must be retried by the host.
193 * What can you do with this hook? Since the SOF signal occurs exactly every
194 * 1 ms (unless the host is in sleep mode), you can use it to tune OSCCAL in
195 * designs running on the internal RC oscillator.
196 * Please note that Start Of Frame detection works only if D- is wired to the
197 * interrupt, not D+. THIS IS DIFFERENT THAN MOST EXAMPLES!
198 */
199#define USB_CFG_CHECK_DATA_TOGGLING 0
200/* define this macro to 1 if you want to filter out duplicate data packets
201 * sent by the host. Duplicates occur only as a consequence of communication
202 * errors, when the host does not receive an ACK. Please note that you need to
203 * implement the filtering yourself in usbFunctionWriteOut() and
204 * usbFunctionWrite(). Use the global usbCurrentDataToken and a static variable
205 * for each control- and out-endpoint to check for duplicate packets.
206 */
207#define USB_CFG_HAVE_MEASURE_FRAME_LENGTH 0
208/* define this macro to 1 if you want the function usbMeasureFrameLength()
209 * compiled in. This function can be used to calibrate the AVR's RC oscillator.
210 */
211#define USB_USE_FAST_CRC 0
212/* The assembler module has two implementations for the CRC algorithm. One is
213 * faster, the other is smaller. This CRC routine is only used for transmitted
214 * messages where timing is not critical. The faster routine needs 31 cycles
215 * per byte while the smaller one needs 61 to 69 cycles. The faster routine
216 * may be worth the 32 bytes bigger code size if you transmit lots of data and
217 * run the AVR close to its limit.
218 */
219
220/* -------------------------- Device Description --------------------------- */
221
222#define USB_CFG_VENDOR_ID (VENDOR_ID & 0xFF), ((VENDOR_ID >> 8) & 0xFF)
223/* USB vendor ID for the device, low byte first. If you have registered your
224 * own Vendor ID, define it here. Otherwise you may use one of obdev's free
225 * shared VID/PID pairs. Be sure to read USB-IDs-for-free.txt for rules!
226 * *** IMPORTANT NOTE ***
227 * This template uses obdev's shared VID/PID pair for Vendor Class devices
228 * with libusb: 0x16c0/0x5dc. Use this VID/PID pair ONLY if you understand
229 * the implications!
230 */
231#define USB_CFG_DEVICE_ID (PRODUCT_ID & 0xFF), ((PRODUCT_ID >> 8) & 0xFF)
232/* This is the ID of the product, low byte first. It is interpreted in the
233 * scope of the vendor ID. If you have registered your own VID with usb.org
234 * or if you have licensed a PID from somebody else, define it here. Otherwise
235 * you may use one of obdev's free shared VID/PID pairs. See the file
236 * USB-IDs-for-free.txt for details!
237 * *** IMPORTANT NOTE ***
238 * This template uses obdev's shared VID/PID pair for Vendor Class devices
239 * with libusb: 0x16c0/0x5dc. Use this VID/PID pair ONLY if you understand
240 * the implications!
241 */
242#define USB_CFG_DEVICE_VERSION 0x00, 0x02
243/* Version number of the device: Minor number first, then major number.
244 */
245#define USB_CFG_VENDOR_NAME 'w', 'i', 'n', 'k', 'e', 'y', 'l', 'e', 's', 's', '.', 'k', 'r'
246#define USB_CFG_VENDOR_NAME_LEN 13
247/* These two values define the vendor name returned by the USB device. The name
248 * must be given as a list of characters under single quotes. The characters
249 * are interpreted as Unicode (UTF-16) entities.
250 * If you don't want a vendor name string, undefine these macros.
251 * ALWAYS define a vendor name containing your Internet domain name if you use
252 * obdev's free shared VID/PID pair. See the file USB-IDs-for-free.txt for
253 * details.
254 */
255#define USB_CFG_DEVICE_NAME 'j', 'j', '4', '0'
256#define USB_CFG_DEVICE_NAME_LEN 4
257/* Same as above for the device name. If you don't want a device name, undefine
258 * the macros. See the file USB-IDs-for-free.txt before you assign a name if
259 * you use a shared VID/PID.
260 */
261/*#define USB_CFG_SERIAL_NUMBER 'N', 'o', 'n', 'e' */
262/*#define USB_CFG_SERIAL_NUMBER_LEN 0 */
263/* Same as above for the serial number. If you don't want a serial number,
264 * undefine the macros.
265 * It may be useful to provide the serial number through other means than at
266 * compile time. See the section about descriptor properties below for how
267 * to fine tune control over USB descriptors such as the string descriptor
268 * for the serial number.
269 */
270#define USB_CFG_DEVICE_CLASS 0
271#define USB_CFG_DEVICE_SUBCLASS 0
272/* See USB specification if you want to conform to an existing device class.
273 * Class 0xff is "vendor specific".
274 */
275#define USB_CFG_INTERFACE_CLASS 3 /* HID */
276#define USB_CFG_INTERFACE_SUBCLASS 1 /* Boot */
277#define USB_CFG_INTERFACE_PROTOCOL 1 /* Keyboard */
278/* See USB specification if you want to conform to an existing device class or
279 * protocol. The following classes must be set at interface level:
280 * HID class is 3, no subclass and protocol required (but may be useful!)
281 * CDC class is 2, use subclass 2 and protocol 1 for ACM
282 */
283#define USB_CFG_HID_REPORT_DESCRIPTOR_LENGTH 0
284/* Define this to the length of the HID report descriptor, if you implement
285 * an HID device. Otherwise don't define it or define it to 0.
286 * If you use this define, you must add a PROGMEM character array named
287 * "usbHidReportDescriptor" to your code which contains the report descriptor.
288 * Don't forget to keep the array and this define in sync!
289 */
290
291/* #define USB_PUBLIC static */
292/* Use the define above if you #include usbdrv.c instead of linking against it.
293 * This technique saves a couple of bytes in flash memory.
294 */
295
296/* ------------------- Fine Control over USB Descriptors ------------------- */
297/* If you don't want to use the driver's default USB descriptors, you can
298 * provide our own. These can be provided as (1) fixed length static data in
299 * flash memory, (2) fixed length static data in RAM or (3) dynamically at
300 * runtime in the function usbFunctionDescriptor(). See usbdrv.h for more
301 * information about this function.
302 * Descriptor handling is configured through the descriptor's properties. If
303 * no properties are defined or if they are 0, the default descriptor is used.
304 * Possible properties are:
305 * + USB_PROP_IS_DYNAMIC: The data for the descriptor should be fetched
306 * at runtime via usbFunctionDescriptor(). If the usbMsgPtr mechanism is
307 * used, the data is in FLASH by default. Add property USB_PROP_IS_RAM if
308 * you want RAM pointers.
309 * + USB_PROP_IS_RAM: The data returned by usbFunctionDescriptor() or found
310 * in static memory is in RAM, not in flash memory.
311 * + USB_PROP_LENGTH(len): If the data is in static memory (RAM or flash),
312 * the driver must know the descriptor's length. The descriptor itself is
313 * found at the address of a well known identifier (see below).
314 * List of static descriptor names (must be declared PROGMEM if in flash):
315 * char usbDescriptorDevice[];
316 * char usbDescriptorConfiguration[];
317 * char usbDescriptorHidReport[];
318 * char usbDescriptorString0[];
319 * int usbDescriptorStringVendor[];
320 * int usbDescriptorStringDevice[];
321 * int usbDescriptorStringSerialNumber[];
322 * Other descriptors can't be provided statically, they must be provided
323 * dynamically at runtime.
324 *
325 * Descriptor properties are or-ed or added together, e.g.:
326 * #define USB_CFG_DESCR_PROPS_DEVICE (USB_PROP_IS_RAM | USB_PROP_LENGTH(18))
327 *
328 * The following descriptors are defined:
329 * USB_CFG_DESCR_PROPS_DEVICE
330 * USB_CFG_DESCR_PROPS_CONFIGURATION
331 * USB_CFG_DESCR_PROPS_STRINGS
332 * USB_CFG_DESCR_PROPS_STRING_0
333 * USB_CFG_DESCR_PROPS_STRING_VENDOR
334 * USB_CFG_DESCR_PROPS_STRING_PRODUCT
335 * USB_CFG_DESCR_PROPS_STRING_SERIAL_NUMBER
336 * USB_CFG_DESCR_PROPS_HID
337 * USB_CFG_DESCR_PROPS_HID_REPORT
338 * USB_CFG_DESCR_PROPS_UNKNOWN (for all descriptors not handled by the driver)
339 *
340 * Note about string descriptors: String descriptors are not just strings, they
341 * are Unicode strings prefixed with a 2 byte header. Example:
342 * int serialNumberDescriptor[] = {
343 * USB_STRING_DESCRIPTOR_HEADER(6),
344 * 'S', 'e', 'r', 'i', 'a', 'l'
345 * };
346 */
347
348#define USB_CFG_DESCR_PROPS_DEVICE 0
349#define USB_CFG_DESCR_PROPS_CONFIGURATION USB_PROP_IS_DYNAMIC
350//#define USB_CFG_DESCR_PROPS_CONFIGURATION 0
351#define USB_CFG_DESCR_PROPS_STRINGS 0
352#define USB_CFG_DESCR_PROPS_STRING_0 0
353#define USB_CFG_DESCR_PROPS_STRING_VENDOR 0
354#define USB_CFG_DESCR_PROPS_STRING_PRODUCT 0
355#define USB_CFG_DESCR_PROPS_STRING_SERIAL_NUMBER 0
356#define USB_CFG_DESCR_PROPS_HID USB_PROP_IS_DYNAMIC
357//#define USB_CFG_DESCR_PROPS_HID 0
358#define USB_CFG_DESCR_PROPS_HID_REPORT USB_PROP_IS_DYNAMIC
359//#define USB_CFG_DESCR_PROPS_HID_REPORT 0
360#define USB_CFG_DESCR_PROPS_UNKNOWN 0
361
362#define usbMsgPtr_t unsigned short
363/* If usbMsgPtr_t is not defined, it defaults to 'uchar *'. We define it to
364 * a scalar type here because gcc generates slightly shorter code for scalar
365 * arithmetics than for pointer arithmetics. Remove this define for backward
366 * type compatibility or define it to an 8 bit type if you use data in RAM only
367 * and all RAM is below 256 bytes (tiny memory model in IAR CC).
368 */
369
370/* ----------------------- Optional MCU Description ------------------------ */
371
372/* The following configurations have working defaults in usbdrv.h. You
373 * usually don't need to set them explicitly. Only if you want to run
374 * the driver on a device which is not yet supported or with a compiler
375 * which is not fully supported (such as IAR C) or if you use a differnt
376 * interrupt than INT0, you may have to define some of these.
377 */
378/* #define USB_INTR_CFG MCUCR */
379/* #define USB_INTR_CFG_SET ((1 << ISC00) | (1 << ISC01)) */
380/* #define USB_INTR_CFG_CLR 0 */
381/* #define USB_INTR_ENABLE GIMSK */
382/* #define USB_INTR_ENABLE_BIT INT0 */
383/* #define USB_INTR_PENDING GIFR */
384/* #define USB_INTR_PENDING_BIT INTF0 */
385/* #define USB_INTR_VECTOR INT0_vect */
386
387/* Set INT1 for D- falling edge to count SOF */
388/* #define USB_INTR_CFG EICRA */
389#define USB_INTR_CFG_SET ((1 << ISC11) | (0 << ISC10))
390/* #define USB_INTR_CFG_CLR 0 */
391/* #define USB_INTR_ENABLE EIMSK */
392#define USB_INTR_ENABLE_BIT INT1
393/* #define USB_INTR_PENDING EIFR */
394#define USB_INTR_PENDING_BIT INTF1
395#define USB_INTR_VECTOR INT1_vect
396
397#endif /* __usbconfig_h_included__ */