summary refs log tree commit diff
path: root/protocol
diff options
context:
space:
mode:
authorBenjamin Gould <bgould@github.com>2014-01-06 14:43:59 -0500
committerBenjamin Gould <bgould@github.com>2014-01-06 17:07:10 -0500
commitf7bb7c4fa99e6bfae311bd6674b6a2bd8537ea73 (patch)
treef5b577d0bf5baaef9c524e9611807451d4f79e40 /protocol
parenta35cf0e458090388202c2a0783c5d34f9b69acf7 (diff)
Added NeXT keyboard converter
Diffstat (limited to 'protocol')
-rw-r--r--protocol/next_kbd.c204
-rw-r--r--protocol/next_kbd.h63
2 files changed, 267 insertions, 0 deletions
diff --git a/protocol/next_kbd.c b/protocol/next_kbd.c
new file mode 100644
index 0000000000..a5a07a7a85
--- /dev/null
+++ b/protocol/next_kbd.c
@@ -0,0 +1,204 @@
+/*
+
+NeXT non-ADB Keyboard Protocol
+ 
+Copyright 2013, Benjamin Gould (bgould@github.com)
+
+Based on:
+TMK firmware code Copyright 2011,2012 Jun WAKO <wakojun@gmail.com>
+Arduino code by "Ladyada" Limor Fried (http://ladyada.net/, http://adafruit.com/), released under BSD license
+
+Timing reference thanks to http://m0115.web.fc2.com/ (dead link), http://cfile7.uf.tistory.com/image/14448E464F410BF22380BB
+Pinouts thanks to http://www.68k.org/~degs/nextkeyboard.html
+Keycodes from http://ftp.netbsd.org/pub/NetBSD/NetBSD-release-6/src/sys/arch/next68k/dev/
+
+This software is licensed with a Modified BSD License.
+All of this is supposed to be Free Software, Open Source, DFSG-free,
+GPL-compatible, and OK to use in both free and proprietary applications.
+Additions and corrections to this file are welcome.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+* Redistributions of source code must retain the above copyright
+  notice, this list of conditions and the following disclaimer.
+
+* Redistributions in binary form must reproduce the above copyright
+  notice, this list of conditions and the following disclaimer in
+  the documentation and/or other materials provided with the
+  distribution.
+
+* Neither the name of the copyright holders nor the names of
+  contributors may be used to endorse or promote products derived
+  from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
+LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGE.
+
+*/
+
+#include <stdint.h>
+#include <stdbool.h>
+#include <util/atomic.h>
+#include <util/delay.h>
+#include "next_kbd.h"
+#include "debug.h"
+
+static inline void out_lo(void);
+static inline void out_hi(void);
+static inline void query(void);
+static inline void reset(void);
+static inline uint32_t response(void);
+
+#define out_hi_delay(intervals)  do { out_hi(); _delay_us(NEXT_KBD_TIMING * intervals); } while (0);
+#define out_lo_delay(intervals)  do { out_lo(); _delay_us(NEXT_KBD_TIMING * intervals); } while (0);
+#define query_delay(intervals)   do { query();  _delay_us(NEXT_KBD_TIMING * intervals); } while (0);
+#define reset_delay(intervals)   do { reset();  _delay_us(NEXT_KBD_TIMING * intervals); } while (0);
+
+void next_kbd_init(void)
+{
+    out_hi();
+    NEXT_KBD_IN_DDR   &= ~(1<<NEXT_KBD_IN_BIT);   // KBD_IN  to input
+    NEXT_KBD_IN_PORT  |=  (1<<NEXT_KBD_IN_BIT);   // KBD_IN  pull up
+    
+    query_delay(5);
+    reset_delay(8);
+    
+    query_delay(5);
+    reset_delay(8);
+}
+
+void next_kbd_set_leds(bool left, bool right)
+{
+    out_lo_delay(9);
+    
+    out_hi_delay(3);
+    out_lo_delay(1);
+    
+    if (left) {
+        out_hi_delay(1);
+    } else {
+        out_lo_delay(1);
+    }
+    
+    if (right) {
+        out_hi_delay(1);
+    } else {
+        out_lo_delay(1);
+    }
+    
+    out_lo_delay(7);
+    out_hi();
+}
+
+#define NEXT_KBD_READ (NEXT_KBD_IN_PIN&(1<<NEXT_KBD_IN_BIT))
+uint32_t next_kbd_recv(void)
+{
+    
+    // First check to make sure that the keyboard is actually connected;
+    // if not, just return
+    // TODO: reflect the status of the keyboard in a return code
+    if (!NEXT_KBD_READ) {
+        sei();
+        return 0;
+    }
+
+    query();
+    uint32_t resp = response();
+    
+    return resp;
+}
+
+static inline uint32_t response(void)
+{
+    cli();
+    
+    // try a 5ms read; this should be called after the query method has
+    // been run so if a key is pressed we should get a response within
+    // 5ms; if not then send a reset and exit
+    uint8_t     i  = 0;
+    uint32_t data  = 0;
+    uint16_t reset_timeout = 50000;
+    while (NEXT_KBD_READ && reset_timeout)  {
+        asm(""); _delay_us(1); reset_timeout--;
+    }
+    if (!reset_timeout) {
+        reset();
+        sei();
+        return 0;
+    }
+    _delay_us(NEXT_KBD_TIMING / 2);
+    for (; i < 22; i++)
+    {
+        if (NEXT_KBD_READ)
+        {
+            data |= ((uint32_t) 1 << i);
+            /* Note:
+             * My testing with the ATmega32u4 showed that there might
+             * something wrong with the timing here; by the end of the
+             * second data byte some of the modifiers can get bumped out
+             * to the next bit over if we just cycle through the data
+             * based on the expected interval.  There is a bit (i = 10) 
+             * in the middle of the data that is always on followed by 
+             * one that is always off - so we'll use that to reset our 
+             * timing in case we've gotten ahead of the keyboard;
+             */
+            if (i == 10)
+            {
+                i++;
+                while (NEXT_KBD_READ) ;
+                _delay_us(NEXT_KBD_TIMING / 2);
+            }
+        } else {
+            /* redundant - but I don't want to remove if it might screw
+             * up the timing
+             */
+            data |= ((uint32_t) 0 << i);
+        }
+        _delay_us(NEXT_KBD_TIMING);
+    }
+    
+    sei();
+    
+    return data;
+}
+
+static inline void out_lo(void)
+{
+    NEXT_KBD_OUT_PORT &= ~(1<<NEXT_KBD_OUT_BIT);
+    NEXT_KBD_OUT_DDR  |=  (1<<NEXT_KBD_OUT_BIT);
+}
+
+static inline void out_hi(void)
+{
+    /* input with pull up */
+    NEXT_KBD_OUT_DDR  &= ~(1<<NEXT_KBD_OUT_BIT);
+    NEXT_KBD_OUT_PORT |=  (1<<NEXT_KBD_OUT_BIT);
+}
+
+static inline void query(void)
+{
+    out_lo_delay(5);
+    out_hi_delay(1);
+    out_lo_delay(3);
+    out_hi();
+}
+
+static inline void reset(void)
+{
+    out_lo_delay(1);
+    out_hi_delay(4);
+    out_lo_delay(1);
+    out_hi_delay(6);
+    out_lo_delay(10);
+    out_hi();
+}
diff --git a/protocol/next_kbd.h b/protocol/next_kbd.h
new file mode 100644
index 0000000000..6d455d4fab
--- /dev/null
+++ b/protocol/next_kbd.h
@@ -0,0 +1,63 @@
+/*
+NeXT non-ADB Keyboard Protocol
+ 
+Copyright 2013, Benjamin Gould (bgould@github.com)
+
+Based on:
+TMK firmware code Copyright 2011,2012 Jun WAKO <wakojun@gmail.com>
+Arduino code by "Ladyada" Limor Fried (http://ladyada.net/, http://adafruit.com/), released under BSD license
+
+Timing reference thanks to http://m0115.web.fc2.com/ (dead link), http://cfile7.uf.tistory.com/image/14448E464F410BF22380BB
+Pinouts thanks to http://www.68k.org/~degs/nextkeyboard.html
+Keycodes from http://ftp.netbsd.org/pub/NetBSD/NetBSD-release-6/src/sys/arch/next68k/dev/
+
+This software is licensed with a Modified BSD License.
+All of this is supposed to be Free Software, Open Source, DFSG-free,
+GPL-compatible, and OK to use in both free and proprietary applications.
+Additions and corrections to this file are welcome.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+* Redistributions of source code must retain the above copyright
+  notice, this list of conditions and the following disclaimer.
+
+* Redistributions in binary form must reproduce the above copyright
+  notice, this list of conditions and the following disclaimer in
+  the documentation and/or other materials provided with the
+  distribution.
+
+* Neither the name of the copyright holders nor the names of
+  contributors may be used to endorse or promote products derived
+  from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
+LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGE.
+
+*/
+
+#include <stdbool.h>
+
+#ifndef NEXT_KBD_H
+#define NEXT_KBD_H
+
+#define NEXT_KBD_KMBUS_IDLE 0x300600
+#define NEXT_KBD_TIMING     50
+
+extern uint8_t next_kbd_error;
+
+/* host role */
+void next_kbd_init(void);
+void next_kbd_set_leds(bool left, bool right);
+uint32_t next_kbd_recv(void);
+
+#endif