Snacks for Bitcoin
Executive Summary
This project is a retrofit of a Crane National 145 Snacktron 1 vending machine, circa 1988, to accept payments and return change in bitcoins. The retrofit takes the form of a Raspberry Pi single-board computer running a software application that communicates with the Bitcoin network over Wi-Fi and with the vending machine via a custom hardware interface module. Bitcoin amounts are translated to and from fiat currency using multiple Internet-accessible price sources.
Hardware Interface

The Raspberry Pi physically and electrically interfaces with the vending machine via a custom hardware module, comprising these parts:
- (1) Custom printed circuit board
- (1) 74AHCT244 integrated circuit, 20-pin PDIP
- Acceptable: Texas Instruments SN74AHCT244N
- (1) 74LVC244A integrated circuit, 20-pin PDIP
- Acceptable: Texas Instruments SN74LVC244AN
- (1) Cinch-Jones 12-pin male plug with clamp
- Acceptable: Molex 38331-5612
- (1) Cinch-Jones 12-pin female socket with clamp
- Acceptable: Molex 38331-8012
- (1) Female pin header, 13×2, 0.1-inch pitch
- Assorted hookup wire, 20 AWG
The vending machine's internal digital-logic signals use 5 volts as the “high” logic level and 0 volts as the “low” logic level. Unfortunately, the Raspberry Pi's GPIO pins cannot tolerate 5-volt inputs, and they produce only 3.3-volt outputs. As the Pi's GPIO pins have a constant input resistance, it would be a simple matter to construct a voltage divider to bring the VMC's 5-volt signals down to 3.3 volts, but this strategy would not work in the opposite direction (from the Pi's 3.3-volt levels to the VMC's 5-volt levels). Amplifying the signals requires transistors. The 74244 series of integrated circuits provides a convenient solution.
The 74244 series is an eight-bit, tri-stateable buffer and line driver. The common 74HC244 requires a VCC (supply voltage) of 5 volts and has a VIH (high-level input voltage) threshold of 3.15 volts at VCC = 4.5V. As this input threshold is dangerously close to 3.3 volts and would be even closer at VCC = 5V, this is too close for comfort. The 74AHCT244 integrated circuit solves this dilemma. Its supply requirements are similar to those of the 74HC244, but despite its VOH (high-level output voltage) being 5 volts, its VIH is only 2 volts, which is safely below the 3.3-volt level of the Pi's GPIO outputs. Thus, the 74AHCT244 is ideal for translating the 3.3-volt signals of the Pi to the 5-volt signals of the vending machine.
Going in the other direction — translating signals from the 5-volt levels of the vending machine to the 3.3-volt levels of the Pi — is achieved conveniently using another variant of the 74244. The 74LVC244A requires a VCC in the range of 1.65 to 3.6 volts, so the 3.3-volt supply on the Pi is suitable. Crucially, the LVC accepts input voltages up to 5.5 volts while its outputs remain clamped at VCC. Thus, powered by the Pi's 3.3-volt supply, this chip efficiently translates from the vending machine's 5-volt signals to the 3.3-volt signals required by the Pi's GPIO inputs.

A custom printed circuit board accommodates the two ICs, routing eight 5-volt signals from one connector grouping to six IC inputs and two IC outputs and another eight 5-volt signals from a second connector grouping to two IC inputs and six IC outputs. The corresponding 3.3-volt IC pins connect to header pins, arranged to mate to the expansion header of the Raspberry Pi. While most of the Pi's GPIO pins can be configured for either input or output, special care is taken to ensure that the Pi's TXD
and RXD
pins, which expose the transmitter and receiver of the Pi's internal UART, are routed respectively to an IC input and IC output, as these two pins are not bi-directional when they are connected to the UART.
The hardware interface module connects the 5-volt supply line from the VMC to the Raspberry Pi's 5-volt supply bus, thereby powering the Pi directly from the vending machine's internal power supply.
Signaling Protocol
The Snacktron implements the MicroMech protocol for interfacing with a coin mechanism. This protocol predates the Multi-Drop Bus (MDB) protocol that is in ubiquitous use today, and details of the MicroMech specification are scant. An extremely poor-quality scan of a pin-out diagram is available in the Coinco 9300-L Operation and Service Manual provided on Coinco's web site. The table of connector pin assignments is reproduced here:
Pin # | Function |
---|---|
1 | 5 VDC Supply Positive |
2 | 5 VDC Supply Return |
3 | Send (0-Volts Active) |
4 | Interrupt (0-Volts Active) |
5 | Data (0-Volts Active) |
6 | Accept Enable (0-Volts Active) |
7 | $.25 Dispense (0-Volts Active) |
8 | $.10 Dispense (0-Volts Active) |
9 | $.05 Dispense (0-Volts Active) |
10 | 117 VDC Supply Return |
11 | Reset (+5VDC Active) |
12 | 117 VDC Supply Positive (rectified unfiltered) |
The hardware interface module positions the Raspberry Pi in-line between the vending machine controller and the coin mechanism. An upstream interface connects the Pi to the VMC, and a downstream interface connects the Pi to the coin mechanism. Ten of the twelve MicroMech signals are present in each of the connector groupings on the interface module. Pins 10 and 12 are absent, as they carry dangerously high voltages. Instead, these pins are connected directly by wire between the upstream and downstream connectors.
The MicroMech signaling protocol is straightforward:
- The VMC asserts
/ACCEPT_ENABLE
when the coin mechanism should accept coins. While this signal is false, the coin mechanism must reject all coins. - The VMC pulses
/DISPENSE25
,/DISPENSE10
, and/DISPENSE5
to actuate the solenoids at the bottom of the coin mechanism's coin tubes to dispense coins. - The VMC pulses
RESET
to reset the internal state of the coin mechanism. - The coin mechanism transmits status, such as notifications of coin acceptance, to the VMC via a handshaking protocol:
- The coin mechanism signals a need to transmit a status byte to the VMC by asserting the
/INTERRUPT
line. - The VMC responds by asserting the
/SEND
line to signal that it is ready to receive. - The coin mechanism transmits a byte on the
/DATA
line using NRZ signaling at 600 baud, least significant bit transmitted first. (This is compatible with the signaling generated and accepted by the UART in the Raspberry Pi.) - The VMC deasserts the
/SEND
line to signal that it has received the byte. - If the VMC needs the same byte transmitted again, it reasserts the
/SEND
line within 4 milliseconds. - Otherwise, the coin mechanism deasserts the
/INTERRUPT
line and considers the byte successfully communicated.
Coin status bytes follow a prescribed format:
D7 |
D6 |
D5 |
D4 |
D3 |
D2 |
D1 |
D0 |
0 |
CVH |
CVL |
/MT05 |
/MT10 |
/MT25 |
0 |
I |
CVH CVL |
Coin Value
|
||||||||
/MT05 |
Nickel Tube Status
|
||||||||
/MT10 |
Dime Tube Status
|
||||||||
/MT25 |
Quarter Tube Status
|
||||||||
I |
Coin Routing
|
Non-coin status bytes take one of a set of prescribed values:
0x03 |
DOLLAR COIN NOT ACCEPTED
A dollar coin was detected but rejected because of prevailing conditions in the acceptor.
|
0x23 |
DOUBLE ARRIVAL
Two coins are presented to the coin acceptor in rapid succession. The acceptor was unable to evaluate either coin. Both coins shall return via the reject slot.
|
0x27 |
COIN JAM
A coin is detected by the coin acceptor as being lodged between the acceptance gates.
|
0x63 |
POWER UP
The coin acceptor has just powered up, or has been RESET, and is ready for coin acceptance.
|
0x67 |
DEFECTIVE SENSOR
The acceptor has sensed a failed coin tube sensor.
|
0x6B |
SLUG
An invalid coin has been detected and returned to the customer via the reject slot.
|
0x6E |
ESCROW RETURN
Indicates the depression of the escrow return lever on the acceptor.
|
0x6F |
NO STROBE
A valid coin has been recognized but not detected passing through to the inventory tubes or to the coin box.
|
Each of the values in the above table shall have 0x10
added to it if the coin mechanism is configured to retain no more than 22 coins in the quarter tube.
Software Application
The Raspberry Pi runs a stock Linux kernel and a privileged daemon in userspace. The daemon monitors and manipulates the Pi's GPIO pins via the standard Linux GPIO Sysfs Interface. Input signal edges are detected in hardware and relayed to userspace via the poll(2)
system call. The daemon is written in C++ and depends on these external libraries:
- zlib – to support reading gzipped responses to HTTP API calls
- GnuTLS – to support making API calls via HTTPS
- GMP – required by the ECDSA math used when signing bitcoin transactions
- libaio – for dispatching overlapped I/O requests to the block layer
The daemon includes a lightweight Bitcoin node implementation that connects to an instance of Bitcoin Core running on any network-accessible machine. Communication is via the Bitcoin P2P protocol, so the Bitcoin Core instance does not need its RPC server enabled. Network traffic is minimal, as the daemon utilizes Bitcoin's Bloom-filter mechanism to filter out almost all transactions except those destined for its own address.
The daemon monitors the bitcoin address associated with the private key supplied to it at startup. When the daemon receives a transaction paying this address, it consults six bitcoin price sources in parallel via public APIs:
If fewer than three of the price sources return a valid response, then the request is retried periodically using a progressive back-off algorithm until at least three of the price sources return valid responses. The daemon respects the Expires
response header and will not request pricing information from a given source while any previously received response from that source remains unexpired.
Once valid pricing data has been acquired, the daemon uses the median price to translate the received bitcoin amount into U.S. cents. It then engages the MicroMech signaling protocol to report to the VMC the sequential insertions of an optimal (fewest-coins) mix of nickels, dimes, quarters, and dollar coins. These coin insertions are reported at a rate of approximately 14 coins per second, limited primarily by the slow baud rate of the data transmissions and by the necessity to wait for the VMC to process each transmission before it signals readiness to accept the next.
After the customer makes a selection using the keypad, the VMC may assert signals to dispense change. The daemon monitors the /DISPENSE5
, /DISPENSE10
, and /DISPENSE25
lines and tallies up the total value of change to return. Once all dispense lines have quiesced for 1 second, the daemon consults the price sources again (respecting any Expires
headers from any previously received responses) and converts the signaled amount of U.S. cents back into an amount of bitcoins. It then builds a bitcoin transaction to pay this amount of bitcoins to the customer, signs the transaction, and transmits it to the connected Bitcoin Core node for dissemination to the Bitcoin network.
The determination of the customer's bitcoin address follows somewhat complex rules:
- If the received payment pays exactly two distinct addresses, one of which necessarily belongs to the daemon, then the other address is assumed to be a change address controlled by the customer, and the daemon will pay any change to this address.
- Otherwise, if the received payment contains an input script that appears to be in the standard form — a DER-encoded ECDSA signature followed by an EC public key — then the public key in the first such input script is assumed to belong to the customer, and the daemon will pay any change to the corresponding P2PKH address.
- Otherwise, the customer's address is undetermined, and the daemon will not pay out any change.
The daemon always constructs change transactions so that they redeem the outputs by which the customer originally paid. This guards against a possible double-spend attack in which an adversary would pay a large amount to the machine while double-spending the coins to himself and would then press the escrow return lever to cause the machine to pay out fresh bitcoins. Because the change payment always depends on the original payment, the change payment will only confirm if the original payment confirms. The customer could still potentially defraud the machine by double-spending to get free snacks, but this is both less likely and less severe.
Since the VMC has a 5-cent resolution on credited amounts, the daemon always rounds incoming payments down to the next 5-cent increment. Thus, some residual value paid by the customer may not be credited to the machine. The daemon remembers this residual value and will add it back in when paying out change.
The daemon supports receiving multiple bitcoin payments for one snack purchase. Change is paid up to the amount of each individual payment output in LIFO order. When constructing change transactions, the daemon pays a mining fee of 1000 satoshis per kibibyte of transaction data (or fraction thereof), and it avoids generating outputs of less than 5460 satoshis (the “dust” threshold).
Building the Daemon
- Install the prerequisites on your Raspberry Pi and on your build machine.
- On your build machine, check out the sources:
$ git clone https://github.com/whitslack/vendingpi.git $ cd vendingpi $ git submodule update --init
- If you have not built the hardware interface module, you may omit the hardware interface code from the compilation:
$ export CPPFLAGS=-DNO_HARDWARE
- Build the daemon:
$ make RPI_REV=2 CHOST=armv6j-hardfloat-linux-gnueabi
- If you are using an original Raspberry Pi (the model with 256 MiB of RAM), set
RPI_REV=1
instead. - If you are compiling on the Raspberry Pi itself, you may omit the
CHOST
variable assignment. Otherwise, adjust it to the cross-compiler prefix appropriate to your system. - If you are omitting the hardware interface code, then you can build the daemon for any ordinary computer (e.g., x86) by compiling on that computer and omitting the
CHOST
. - Copy
out/armv6j-hardfloat-linux-gnueabi/vending
to your Raspberry Pi. - Start the daemon:
$ ./vending --testnet ${BITCOIND_HOST} ${BITCOIND_PORT} ${PRIVKEY}
- Substitute appropriate values for
BITCOIND_HOST
,BITCOIND_PORT
, andPRIVKEY
. - Omit
--testnet
in production.
For demonstration and testing purposes, the daemon supports an extremely rudimentary command interface on standard input:
- Input a positive integer to simulate reception of a bitcoin payment equivalent to the specified number of U.S. cents. If the VMC is connected, the daemon will transmit the appropriate signals to credit the customer with the specified amount (rounded down to the nearest 5-cent increment). No customer bitcoin address will be recorded, so returning change will not be possible.
- Input
0
to simulate actuation of the escrow return lever on the coin mechanism. This is useless without the VMC connected. - Input a negative integer to simulate reception of change-dispensing signals from the VMC totaling the specified (negated) number of U.S. cents. If a bitcoin payment had been received previously, then this will cause a change transaction to be signed and transmitted.
Future Work
The daemon aborts if it gets disconnected from the Bitcoin Core node (or if it is unable to connect at startup). A reconnection scheme with progressive back-off will be implemented to address this fragility.— Implemented inc8250bd
.The daemon does not communicate with the coin mechanism at all. Future work will make use of the hardware interface module's downstream interface to enable simultaneous use of legacy currency (coins) and bitcoins.— Implemented inc02edc6
.Significant time elapses between application of power to the Pi and startup of the daemon. During this time, all of the Pi's GPIO pins except— Implemented in hardware.TXD
are configured as inputs, meaning their signal levels are floating. Some of these pins are connected to inputs on the 74AHCT244 IC, and without a defined voltage level, the IC may drive the dispense lines low, causing the coin mechanism to energize its dispensing solenoids and pay out a coin erroneously. Moreover, these solenoids are specified to be energized for only 100 milliseconds at a time, with a 400-millisecond cool-down period, so keeping them energized during the entire boot process is unacceptable. To address this problem, pull-up resistors will be added to the 74AHCT244 input pins to pull them high until the daemon takes control.- The daemon loses synchronization with the block chain if the Bitcoin Core node switches to a different block chain branch and then switches back. This does not cause a loss of service, but it does mean that synchronizing at the next reboot will take longer. Future work will introduce a block header cache to allow the daemon to switch back to a previously seen chain.