How many times have you needed the next prime number in a sequence and, like some animal, had to go to a printed table to look it up. Well, those days are over.
----
A prime number is any positive whole number that can only get evenly divided only by 1 and itself. Primes are used in many applications; a popular use being for encryption and cryptography. Demonstrated here is another use for prime numbers. That is making use of an older/slower Raspberry PI and a few parts to nerd up the decor of any room.
A few Raspberry PI skills learned will be:
- writing to text files
- reading from text files
- driving a low cost I2C LCD display
- driving a relay via a transistor
- simple graceful shutdown method for the RasPI with a button and a JST connector.
-----
-----
The python source is pretty straight forward. On button press a pointer to a file containing the first few million primes is indexed and displayed on the LCD. A 5VDC relay clicks to represent the non primes in between. The rig runs via USB power and the last found prime is always saved. A handy 'shutdown' button is incorporated to allow the Raspbery PI project to be powered down gracefully if it needs to be moved.
-----
You'll need a Raspberry PI, 5VDC relay, PN2222A transistor, 16x2 I2C LCD, two resistors, and two normally open button switches. A project box holds it all together. Connect it all up like this:
Once on the breadboard it will look a bit like this:
Set up the RasPI to run the python code below at reboot (use @reboot in the sudo crontab).
Note, two files are expected to be found in the working directory:
- prime_list.txt (List of prime numbers in order. One per line. As many as you like)
- high_prime.txt (Holds the highest prime found in case of a restart/reboot)
#!/usr/bin/python
#
# WhiskeyTangoHotel.Com
# OCT 2016
#
# Program reads a file and displays prime # on LCD
# On button press find next prime #. Click a relay for each non prime
# Write new highest prime to file to save if restarting the program
#
# Gives an audible (relay click) representation of the distance between primes
#
# Expects program to be in: /home/pi/RasPI/Programs-RasPI/Prime_Relay/
#
# Leverages LCD script using I2C backpack for 16x2 and 20x4 screens.
# Thanks to Matt Hawkins http://www.raspberrypi-spy.co.uk/
#
#--------------------------------------
import smbus # for LCD I2C display
import time # fpr sleep and pause delays
import os # this is for the shutdown button press
import RPi.GPIO as GPIO
GPIO.setmode(GPIO.BOARD) # to use Raspberry Pi board pin numbers
# set up GPIO output channel
Relay_Pin = 11 # pin to drive the relay clicks via 2222a transistor
GPIO.setup(Relay_Pin, GPIO.OUT)
Relay_delay = .1 # time delay between relay clicks
Switch = 8 # Button switch to move to next prime number
GPIO.setup(Switch, GPIO.IN)
reboot_pin = 26 # Push this button and the RasPI shuts down gracefully
#Set pin to input and set pull-up resistor to hold the pin is high
GPIO.setup(reboot_pin, GPIO.IN, pull_up_down=GPIO.PUD_UP)
#######################################
### LCD DRIVER FUNCTIONS START HERE ###
#######################################
# Define some device parameters
I2C_ADDR = 0x27 # I2C device address
LCD_WIDTH = 16 # Maximum characters per line
# Define some device constants
LCD_CHR = 1 # Mode - Sending data
LCD_CMD = 0 # Mode - Sending command
LCD_LINE_1 = 0x80 # LCD RAM address for the 1st line
LCD_LINE_2 = 0xC0 # LCD RAM address for the 2nd line
LCD_LINE_3 = 0x94 # LCD RAM address for the 3rd line
LCD_LINE_4 = 0xD4 # LCD RAM address for the 4th line
LCD_BACKLIGHT = 0x08 # On
#LCD_BACKLIGHT = 0x00 # Off
ENABLE = 0b00000100 # Enable bit
# Timing constants
E_PULSE = 0.0005
E_DELAY = 0.0005
#Open I2C interface
#bus = smbus.SMBus(0) # Rev 1 Pi uses 0
bus = smbus.SMBus(1) # Rev 2 Pi uses 1 RasPI(Bob)
def lcd_init():
# Initialise display
lcd_byte(0x33,LCD_CMD) # 110011 Initialise
lcd_byte(0x32,LCD_CMD) # 110010 Initialise
lcd_byte(0x06,LCD_CMD) # 000110 Cursor move direction
lcd_byte(0x0C,LCD_CMD) # 001100 Display On,Cursor Off, Blink Off
lcd_byte(0x28,LCD_CMD) # 101000 Data length, number of lines, font size
lcd_byte(0x01,LCD_CMD) # 000001 Clear display
time.sleep(E_DELAY)
def lcd_byte(bits, mode):
# Send byte to data pins
# bits = the data
# mode = 1 for data
# 0 for command
bits_high = mode | (bits & 0xF0) | LCD_BACKLIGHT
bits_low = mode | ((bits<<4) & 0xF0) | LCD_BACKLIGHT
# High bits
bus.write_byte(I2C_ADDR, bits_high)
lcd_toggle_enable(bits_high)
# Low bits
bus.write_byte(I2C_ADDR, bits_low)
lcd_toggle_enable(bits_low)
def lcd_toggle_enable(bits):
# Toggle enable
time.sleep(E_DELAY)
bus.write_byte(I2C_ADDR, (bits | ENABLE))
time.sleep(E_PULSE)
bus.write_byte(I2C_ADDR,(bits & ~ENABLE))
time.sleep(E_DELAY)
def lcd_string(message,line):
# Send string to display
message = message.ljust(LCD_WIDTH," ")
lcd_byte(line, LCD_CMD)
for i in range(LCD_WIDTH):
lcd_byte(ord(message[i]),LCD_CHR)
#######################################
### LCD DRIVER FUNCTIONS END HERE ###
#######################################
# Highest calulated prime is stored in a file. Get that number
highprime = open('/home/pi/RasPI/Programs-RasPI/Prime_Relay/high_prime.txt','r')
storedprime = (highprime.readline())
storedprime_val = int(storedprime)
print ' '
print "Recalled highest prime from file is: " + storedprime
print ' '
#Put status on LCD
lcd_init()
lcd_string("Init to last",LCD_LINE_1)
lcd_string("prime: " + str(storedprime_val),LCD_LINE_2)
highprime.close
def main():
# Main program block
# Relay test clicks
for i in range(1, 3):
GPIO.output(Relay_Pin,GPIO.HIGH) # Close relay
time.sleep(Relay_delay)
GPIO.output(Relay_Pin,GPIO.LOW) # Open relay
time.sleep(Relay_delay)
# Set the pointer to the correct place in the file that holds the prime list
# prime_list.txt contains primes to 1,299,709
primefile = open('/home/pi/RasPI/Programs-RasPI/Prime_Relay/prime_list.txt','r')
currentprime_val = 0
while storedprime_val > currentprime_val:
currentprime = (primefile.readline())
currentprime_val = int(currentprime)
# Update the LCD
lcd_string("PRESS for next",LCD_LINE_1)
lcd_string("prime: " + str(currentprime_val),LCD_LINE_2)
time.sleep(0.5)
# Relay test clicks
for i in range(1, 3):
GPIO.output(Relay_Pin,GPIO.HIGH) # Close relay
time.sleep(Relay_delay)
GPIO.output(Relay_Pin,GPIO.LOW) # Open relay
time.sleep(Relay_delay)
while True:
if (GPIO.input(reboot_pin) == 0):
# Update the LCD
lcd_string("Primes in a Box",LCD_LINE_1)
lcd_string("shutting down!!!" + str(currentprime_val),LCD_LINE_2)
time.sleep(5)
#Send command to system to shutdown
os.system("sudo shutdown -h now")
if (GPIO.input(Switch) == 0):
# Update the LCD
lcd_string("Calculating next",LCD_LINE_1)
lcd_string("prime: " + str(currentprime_val),LCD_LINE_2)
oldprime = currentprime_val
currentprime = (primefile.readline())
currentprime_val = int(currentprime)
deltaprime = currentprime_val - oldprime
for i in range(1, deltaprime+1): # click relay for each non prime between primes
#print i
GPIO.output(Relay_Pin,GPIO.HIGH) # Close relay
time.sleep(Relay_delay)
GPIO.output(Relay_Pin,GPIO.LOW) # Open relay
time.sleep(Relay_delay)
#Write the prime to a file so it can be recalled at prog start
hp = open("/home/pi/RasPI/Programs-RasPI/Prime_Relay/high_prime.txt","w")
hp.write(str(currentprime_val))
print "CURRENT " + str(currentprime_val)
print " OLD " + str(oldprime)
print '------- '
print " DELTA " + str(deltaprime)
#print i
print ' '
# Update the LCD
lcd_string("PRESS for next",LCD_LINE_1)
lcd_string("prime: " + str(currentprime_val),LCD_LINE_2)
hp.close
time.sleep(.250)
if __name__ == '__main__':
try:
main()
except KeyboardInterrupt:
pass
finally:
lcd_byte(0x01, LCD_CMD)
----Thanks for the visit and happy prime numbering!
