Skip to content

Latest commit

 

History

History
413 lines (283 loc) · 25 KB

README.md

File metadata and controls

413 lines (283 loc) · 25 KB

Writing Software for the Kindle Keyboard

It might be 9 years old, but the Kindle Keyboard is a pretty neat device. It runs Linux, has a 600x800 eink Pearl screen that can display 16 shades of grey, and has a small keyboard, five-point control, page buttons, and a speaker.

It also turns out that you can compile Go code for the device using the flags env GOOS=linux GOARCH=arm GOARM=6 go build program.go.

Background info is in paragraphs; steps I take will be in unordered lists.

Day 1

The first day. I spent a bunch of time doing research and seeing what options there for writing software for the Kindle 3.

There was a Java SDK called the Kindle Development Kit or KDK) released by Amazon not long after the device was released, but it was EOL'ed in 2013 and is not easily available. The device runs Java 1.4, which is pretty old.

The central place for finding out information about Kindle development is the Kindle Developer's Corner on the MobileRead forums. Interestingly, much of the code that folks have written hasn't made it to Github or other code sharing sites.

Most third-party code is distributed in the same format as official Kindle firmware updates by placing it in the root of the Kindle USB device and restarting the Kindle. The format of these is some kind of obfuscated tar file.

Someone made a neat weather display that is nice inspiration.

I have seen many links to this site that describes extracting the Jars from the Kindle and writing a Java app using the KDK.

Day 2

I need to be able to get a shell and poke around. I'm a little worried as this thing is almost a decade old, and I'm not sure what will be run on it. Evidently it's libc is from 2006.

There is a command called eips that will perform various functions with the screen, but all I've been able to do is get it to clear the screen using eips -c and print info about the screen using eips -i.

  • I tried to display a PNG, and something happened, but the image only partially was displayed and was stretched. This image from the weather project, however, worked, so it's possible. There is probably something wrong with the image format I am using.

Day 3

Today I wanted to try to draw something to the screen.

  • Disabled built-in kindle UI with:/etc/init.d/framework stop. I also looked around at the other stuff in /etc/init.d.

I looked back at the weather display project and noticed that the PNGs need to be created with a color_type value in the PNG header to 0. pngcrush -c 0 will do this.

  • I was able to display arbitrary graphics on the device using eips -g!

I had previous cross-compiled Go and run it on other ARM devices, so I figured it was worth a shot.

  • I compiled the example program from this article:

    package main
    
    import "fmt"
    import "runtime"
    
    func main() {
            fmt.Printf("Hello %s/%s\n", runtime.GOOS, runtime.GOARCH)
    }

    env GOOS=linux GOARCH=arm go build test.go produced an executable that resulted in an Illegal instruction when run on the Kindle. Adding GOARM=6 fixed this, though, and allowed the program to run!

  • So far, compiling on the device, scping the program over to the Kindle, and running it in SSH session has been working pretty well. At some point it would be nice to automate this.

Day 4

Today I wanted to draw some graphics. The Kindle Keyboard has a framebuffer device available at /dev/fb0, so that is where I'm going to start. Once I saw that Go would run on the device, I became a lot less interested in using Java and the KDK.

A 4-bit number will hold values from 0-15, which is all that's needed to represent one of the 16 levels of grey supported by the display.

The framebuffer on the Kindle packs two 4-bit pixels into each byte. This means that each row of the display uses 300 bytes to hold the values of its 600 pixels. This is enough information to start working with the raw framebuffer data. Writing to a pixel will involve setting either the higher or lower 4 bits of the appropriate byte to a value representing the grey value to display.

  • I tried using some code I had written for another experiment to draw to the framebuffer, leveraging this framebuffer library. Unfortunately, I ran into an issue because the value of Smem_len is zero, so this library thinks there is nowhere to write data to. I hardcoded 483328 (which I got from running eips -i), which allowed something to be drawn to the screen. Looking at what I would need to do to modify that library to handle 4-bit pixels packed two to a byte instead of 4-byte RGBA pixels, I decided it would be easy enough to Mmap the framebuffer myself and write a few utility functions. I'm pretty sure that I only need to map 240,000 bytes to do what I need to do, and since this code is only ever going to run on this device (any maybe a DX if I find one), hardcoding these values is easier than troubleshooting why the FBIOGET_FSCREENINFO ioctl syscall isn't returning the expected value for Smem_len.
  • After writing to the framebuffer, you have to tell the device to update the display. So far I have been using echo 1 > /proc/eink_fb/update-display to do this. There is probably a better way.
  • I was able to draw some simple graphics on the display at the end of this session.

Day 5

Today's another day for research and poking around.

/etc/init.d/battcheck returns a bunch of interesting battery statistics:

system: I battcheck:def:running
Sat Jan 19 19:52:48 2019  INFO:battery voltage: 4136 mV
Sat Jan 19 19:52:48 2019  INFO:battery charge: 86%
system: I battcheck:def:current voltage = 4133mV
Sat Jan 19 19:52:48 2019  INFO:battery charge: 86%
Sat Jan 19 19:52:48 2019  INFO:battery voltage: 4136 mV
Sat Jan 19 19:52:48 2019  INFO:battery current: 337 mA
system: I battcheck:def:gasgauge capacity=86% volts=4136 mV current=337 mA
system: I battcheck:def:Waiting for 3460mV or 4%
system: I battcheck:def:battery sufficient, booting to normal runlevel

Here's the output of lsmod:

Module                  Size  Used by
ar6000                161076  0
g_ether                21096  0
eink_fb_shim          116732  0
eink_fb_hal_broads    397532  0
eink_fb_hal            59764  5 eink_fb_shim,eink_fb_hal_broads
volume                  8900  0
fiveway                23552  0
mxc_keyb               15904  0
uinput                  7776  0
fuse                   48348  2
arcotg_udc             38628  1 g_ether
mwan                    7324  0

There isn't a tree command on the Kindle, and I've been using ls -R a lot to explore the filesystem. I'm considering scping the entire disk to the Mac so I can use my usual tools on it.

The Kindle is running alsa version 1.0.13:

$ alsactl -v
alsactl version 1.0.13`

To run alsamixer, TERM needs to be set to xterm (mine was xterm-256color).

Day 6

I spent some timing looking at graphics packages, with an eye toward rendering text to the screen in a scalable way. I originally planned to use bitmap fonts due to their ease of use, but then I found a pure-Go implementation of freetype, and then, a while later, gg, which provides a nice API for drawing graphics and text. It even includes wrapping text to lines, which is something that isn't provided by freetype.

man isn't available on the Kindle, which makes figuring out the arguments for the old versions of everything a little more challenging.

One of the twists of working with the eink display is that values that would appear on light-emitting displays as dark colors instead appear as light colors on eink displays. This in effect inverses everything, which needs to be compensated for somewhere in the graphics stack of a program.

  • Wrote a simple program circle to draw a black circle on the center of the Kindle screen, centering the string Hello! within it. There are 7 concentric circles with decreasing stroke width surrounding it.

  • Finally setup ssh keys so I could ssh into the Kindle without hitting enter at the password prompt by following these directions. I copied the public key over using scp kindle_rsa.pub [email protected]:/mnt/us/usbnet/etc/authorized_key.

  • Added build_and_run, a script to automate the process of compiling a program, copying it to the Kindle, and running it.

In the back of my mind I was a little worried about memory on the Kindle, but it's looking like things are going to be just fine:

             total       used       free     shared    buffers     cached
Mem:           250        219         30          0         82         97
-/+ buffers/cache:         40        210
Swap:            0          0          0

Day 7

  • Added a new executable, screengrab, that generates a PNG from the current state of the framebuffer. Wrapped this in a script to capture a screenshot on the device, scp it back to the host, and open it in the default program.
  • Moved all scripts into scripts directory to keep the root of the repo tidy.

Day 8

  • I attempted to play a variety of sound files, including those I found on the Kindle via aplay, but all that came out of the speaker was a deafening static that came and went in waves. Not completely unlike an ocean, really, but also fairly unpleasant and not at all resembling the piano music I was hoping to hear. This page, however, prompted me to attempt to play a file that was created on the device using arecord, which worked! So converting audio to the format of that file should allow for custom sound to be played. 8khz mono isn't going to sound great, but it's better than nothing.

  • I later learned that the issue is that aplay expects raw audio data (say, from a WAV file). A 44.1khz stereo WAV played fine. I was previously trying to play an MP3, which isn't raw audio data and needs to be decoded first.

gasgauge returns stats related to the battery and charging status of the device:

[root@kindle root]# gasgauge-info -c
100%
Tue Jan 22 03:01:11 2019  INFO:battery charge: 100%

The Kindle has a say command that will speak arbitrary text (source). I confirmed that this works!

evtest is available on the Kindle, which makes exploring the hardware keyboard pretty straight forward. The main keyboard, five-way control, and paging buttons are all seperate devices.

  • Wrote a small program to read key events from the main keyboard (/dev/input/event0) and print the raw bytes to the screen:
    $ script/build_and_run keys
    ...16 [71 174 70 92 195 96 12 0 1 0 52 0 1 0 0 0]
    16 [72 174 70 92 232 199 0 0 1 0 52 0 0 0 0 0]
    16 [72 174 70 92 31 218 10 0 1 0 52 0 1 0 0 0]
    16 [72 174 70 92 232 252 12 0 1 0 52 0 0 0 0 0]
    16 [72 174 70 92 175 170 14 0 1 0 52 0 1 0 0 0]
    16 [73 174 70 92 24 61 1 0 1 0 52 0 0 0 0 0]
    

Day 9

The Kindle Keyboard Linux kernel is 2.6.26:

[root@kindle root]# uname -r
2.6.26-rt-lab126

This site is super useful for looking up the definitions of input_type on a specific version of the Linux kernel, which I need to do in order to load the raw data I'm receiving from /dev/input/event0 into a Go struct.

  • Wrote code to handle processing the events coming from /dev/input/event0 (the main keyboard) and push Go structs representing them onto a channel. Used stringer to generate the String() method for these types.

The Kindle Keyboard hardware or drivers (not sure which) are interesting. Look at what events are sent when the the shift key is pressed and released, followed by the z key:

{Time:2019-01-23 03:30:09.780995 +0015 GMT-00:20 Type:KeyDown Key:KeyShift}
{Time:2019-01-23 03:30:09.871003 +0015 GMT-00:20 Type:KeyUp Key:KeyShift}
{Time:2019-01-23 03:30:10.70096 +0015 GMT-00:20 Type:KeyDown Key:KeyZ}
{Time:2019-01-23 03:30:10.860955 +0015 GMT-00:20 Type:KeyUp Key:KeyZ}

Compare this to the same thing, but for the alt key:

{Time:2019-01-23 03:30:15.191005 +0015 GMT-00:20 Type:KeyDown Key:KeyAlt}
{Time:2019-01-23 03:30:15.191294 +0015 GMT-00:20 Type:KeyDown Key:KeyZ}
{Time:2019-01-23 03:30:15.191521 +0015 GMT-00:20 Type:KeyUp Key:KeyZ}
{Time:2019-01-23 03:30:15.191528 +0015 GMT-00:20 Type:KeyUp Key:KeyAlt}

Notice how the alt KeyUp event isn't sent until after another key is pressed, which is the same thing you see if the modifier was held down by the user:

{Time:2019-01-23 03:30:17.881007 +0015 GMT-00:20 Type:KeyDown Key:KeyShift}
{Time:2019-01-23 03:30:18.180979 +0015 GMT-00:20 Type:KeyDown Key:KeyZ}
{Time:2019-01-23 03:30:18.400987 +0015 GMT-00:20 Type:KeyUp Key:KeyZ}
{Time:2019-01-23 03:30:18.630976 +0015 GMT-00:20 Type:KeyUp Key:KeyShift}

{Time:2019-01-23 03:30:22.240992 +0015 GMT-00:20 Type:KeyDown Key:KeyAlt}
{Time:2019-01-23 03:30:22.241281 +0015 GMT-00:20 Type:KeyDown Key:KeyZ}
{Time:2019-01-23 03:30:22.400988 +0015 GMT-00:20 Type:KeyUp Key:KeyZ}
{Time:2019-01-23 03:30:22.700979 +0015 GMT-00:20 Type:KeyUp Key:KeyAlt}

This also means that (at least using the technique I am and listening to dev/input/event0) it's impossible to detect a keypress of only the alt key.

  • Wrote a utility program, simulate_eink, to convert a PNG by mapping the shades of gray to a pallete that is perceptually much closer to what the eink screen looks like to a human. It also adds a bit of random noise for realism.

  • Updated the draw command to use the latest FrameBuffer code.

  • Added example images for draw and circle.

Day 10

Today I wanted to run the built-in say whenever a key on the keyboard was pressed. Unfortunately, I hit an issue with using exec.Command:

$ script/build_and_run letters
...Q
goroutine 1 [running]:
runtime/debug.Stack(0x1045a000, 0xe8fa0, 0x104481e0)
	/usr/local/Cellar/go/1.10.2/libexec/src/runtime/debug/stack.go:24 +0x80
main.main()
	/Users/jimb/go/src/github.com/jim/kindleland/cmd/letters/letters.go:25 +0x124

panic: fork/exec /usr/bin/say: function not implemented

goroutine 1 [running]:
main.main()
	/Users/jimb/go/src/github.com/jim/kindleland/cmd/letters/letters.go:26 +0x230

After some searching online, I was worried that exec.Command might be relying on glibc, which the Kindle has an ancient version of:

[root@kindle root]# /lib/libc.so.6
GNU C Library stable release version 2.5, by Roland McGrath et al.
Copyright (C) 2006 Free Software Foundation, Inc.
This is free software; see the source for copying conditions.
There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A
PARTICULAR PURPOSE.
Compiled by GNU CC version 4.1.2.
Compiled on a Linux 2.6.15 system on 2008-06-10.
...

That software was compiled over a decade ago. And indeed there are threads about Golang not running on old versions of glibc, and I couldn't find a specific glibc version requirement, but I wanted to get a better look at what was going on. To examine what syscalls the program was making and what/which errors it was getting back, I ran the program with strace and saw the following:

pipe2(0x10431dac, O_CLOEXEC)            = -1 ENOSYS (Function not implemented)
pipe2(0x10431dac, 0)                    = -1 ENOSYS (Function not implemented)

The pipe2 syscall was added in Linux 2.6.27, and the Kindle has 2.6.23 (which is currently listed as Go's minimum supported Linux version).

os.Pipe is called several layers within the exec.Command code when Command.Run() is executed. There is a fallback in the code to handle pipe2 not being supported, and you can see how the calls to syscall.Pipe2 and syscall.Pipe map to the two syscalls shown above.

// Pipe returns a connected pair of Files; reads from r return bytes written to w.
// It returns the files and an error, if any.
func Pipe() (r *File, w *File, err error) {
	var p [2]int

	e := syscall.Pipe2(p[0:], syscall.O_CLOEXEC)
	// pipe2 was added in 2.6.27 and our minimum requirement is 2.6.23, so it
	// might not be implemented.
	if e == syscall.ENOSYS {
		// See ../syscall/exec.go for description of lock.
		syscall.ForkLock.RLock()
    e = syscall.Pipe(p[0:])
    ...

Except, of course, that the second syscall should be pipe, not pipe2 based on the fallback code. The problem is that syscall.Pipe is implemented using the pipe2 syscall on linux/arm! This change happened here.

By restoring the previous definition of syscall.Pipe in syscall/syscall_linux_arm.go, I was able to get my code that uses exec.Command to run properly on the Kindle.

In the process of working through this, I attemped to compile Delve for some on-device debugging before discovering that Delve doesn't support ARM.

And of course, after digging into these syscalls, I read that you can also just write text to /var/tmp/ttsUSFifo to use the Kindle's text to speech.

  • Wrote a little program, speak, that uses say to speak the name of each key as it is pressed.

Day 11

I'm having an issue using gg to render text to an image and display it on the screen. There is a memory leak somewhere, and after a certain number of updates the devices bogs down and things get weird. I am going to do some troubleshooting and see if the issue is in my code and in the way I am reusing the gg.Context multiple times instead of creating a new one each time I want to draw something.

FBink is a C library that does a lot of what I want to do. It was originally designed for the Kobo but now also supports Kindles and other Eink devices (they tend to have very similar hardware and software stacks). There are Go bindings for it, but to use them you have to enable cgo, which is something I would like to avoid to keep the build process as simple as possible. However, these libraries are excellent references as I move forward with a pure-Go approach.

Specifically interesting are the parts of FBInk that expose how to update only part of the screen. So far I have been doing entire device updates, which are slow and provide a jarring experience for the user.

Amazon posts the source code they are required to release for all Kindle devices and apps. linux-2.6.26/include/linux/einkfb.h shows a lot of the details used by FBInk to do its work (and is actually used directly in that project).

Day 12

Today was a day spent learning about cgo, linux headers, and go generate.

go tool cgo -godefs ignores the special // #cgo comments, so options that need to be passed to the C compiler have to be passed on the command line. The docs aren't super clear about this, but I was able to sort it out by using the -gcc-debug flag to cgo and then running that output through clanng myself, adding the -v flag so I could experiment with which options needed to be passed to get the lookup paths correct.

I ended up using the following to generate Go code from the einkfb.h file included in the GPL source distribution for the Kindle:

go tool cgo -godefs -- -Ivendor/linux-2.6.26-lab126/include -D__KERNEL__ constant_defs.go > constants.go

By putting this line in a shell script, script/generate_constants, I was able to invoke it by adding a special comment to constant_defs.go and then running go generate.

Day 13

I went down a rabbit hole learning more about linux syscalls, ioctls, and how to interact with them in Go. I got the basic screen update working by making an IOCTL call on /dev/fb/0 and was able to specify "fast" or "slow", although I haven't yet sorted out what the difference between the different options are.

My next task is to sort out how to do a partial screen update. To do so, I need to pass a pointer to a struct into the syscall which contains information about how to do the update: what areas to (not) update, what FX to use, etc. There are a few levels of software running on the Kindle 3, which makes keeping everything straight a little bit harder.

This article was the best thing I found while trying to figure out how I would pass an area of the screen to update.

It appears that cgo -godefs doesn't support most macros, which makes the way I was trying to define the constants from a pervious day a dead end as the header I am working with includes a lot of stuff like this:

#define FBIO_EINK_UPDATE_DISPLAY            _IO(FBIO_MAGIC_NUMBER, 0xdb) // 0x46db (fx_type)
#define FBIO_EINK_UPDATE_DISPLAY_AREA       _IO(FBIO_MAGIC_NUMBER, 0xdd) // 0x46dd (update_area_t *)

I am probably just going to define the values I need in Go as they come up instead of attempting to autogenerate things from the C header. godefs may prove to be a useful tool, though, because it will automate the conversion when I need new values.

I also saw that the built-in syscall package is considered deprecated and that you are supposed to use sys instead.

Day 14

Today I decided to try to use the FBInk library a try. I did end up getting it to compile in a Docker container, although it took a lot of time to get everything working and for the cross-compiling toolchain it expects to install itself. I'm still pretty sold on avoiding C, but I am close to being able to use this library to see if that is useful.

FROM ubuntu:latest
RUN apt-get update
RUN apt-get -y install gperf help2man bison texinfo flex gawk git build-essential autoconf libncurses5-dev curl wget file
WORKDIR /root/src
RUN git clone https://github.com/koreader/koxtoolchain.git
WORKDIR /root/src/koxtoolchain
ENV CT_EXPERIMENTAL=y
ENV CT_ALLOW_BUILD_AS_ROOT=y
ENV CT_ALLOW_BUILD_AS_ROOT_SURE=y
RUN ./gen-tc.sh kindle
WORKDIR /root/src
RUN git clone https://github.com/NiLuJe/FBInk
WORKDIR /root/src/FBInk
RUN git submodule update --init

Day 15

Spent some time getting the various screen update functions to work from Go without using FBInk. Ran strace eips -c to see how that program cleared the screen. The interesting part:

...
open("/dev/fb/0", O_RDWR)               = 3
ioctl(3, FBIOGET_VSCREENINFO or PF_IOCTL_INIT, 0xbeb52ae0) = 0
mmap2(NULL, 240000, PROT_READ|PROT_WRITE, MAP_SHARED|MAP_LOCKED, 3, 0) = 0x40146000
msync(0x40146000, 240000, MS_SYNC)      = 1
ioctl(3, FBIO_EINK_CLEAR_SCREEN, 0)     = 0
close(3)
...

It's good to see that program is also mmap2ing 240000 bytes, just as I am.

It seems that you need to clear the screen twice to really get a clean slate.

I also discovered that the main keyboard send different keycodes when Alt is combined with the top row of letter keys:

# Alt-Q
{Time:2019-02-06 02:13:56.412672 +0015 GMT-00:20 Type:KeyDown Key:KeyType(2)}
{Time:2019-02-06 02:13:56.57269 +0015 GMT-00:20 Type:KeyUp Key:KeyType(2)}

# Alt-P
{Time:2019-02-06 02:14:01.742672 +0015 GMT-00:20 Type:KeyDown Key:KeyType(11)}
{Time:2019-02-06 02:14:01.892816 +0015 GMT-00:20 Type:KeyUp Key:KeyType(11)}

Day 16

  • Wrote a small program that used the freetype package to draw text to the screen.

Day 17

  • Improved text program to have it wrap text at the end of a line and draw within a defined part of the screen.

Day 18

  • Added a new letters program that allows large letters to be typed across the screen. It becomes sluggish when many keys are pressed quickly. I will need to add some throttling to the screen updating and move updating the buffer and telling the screen to refresh to a goroutine.