Since finding a cheap alternative to PLC whilst using an industrial protocol is a popular idea the Raspberry Pi has caught many eyes on doing this. I posted once on reading a temperature sensor and serving it up on the Pi using ModbusTCP. This time I expound on it and show you how to control something. In this case it will be a PC 12v fan.
Parts:
- 1 x Raspberry pi
- 1 x Darlington Transistor
- 1 x PC Fan (with about 500ma load)
- 1 x 12v DC Power Supply
Setup:
- Installing pymodbus and dependencies:
- sudo apt-get install python-pymodbus python-twisted-conch
- Wire up your Fan, LED, or motor
- View my code from Github.
Video of what I did:
Source Code:
from pymodbus.server.async import StartTcpServer
from pymodbus.device import ModbusDeviceIdentification
from pymodbus.datastore import ModbusSequentialDataBlock, ModbusSlaveContext, ModbusServerContext
from pymodbus.transaction import ModbusRtuFramer, ModbusAsciiFramer
from twisted.internet.task import LoopingCall
from threading import Thread
import pid
import threading
from time import sleep
import RPi.GPIO as GPIO
import os
os.system('modprobe w1-gpio')
os.system('modprobe w1-therm')
#set up Raspberry GPIO
GPIO.setmode(GPIO.BCM)
GPIO.setwarnings(False)
GPIO.setup(25,GPIO.OUT)
pwm = GPIO.PWM(25,60)
pwmDutyCycle=100
pwm.start(pwmDutyCycle)
temperaturePoll = None
class Temp(Thread):
"""
A class for getting the current temp of a DS18B20
"""
def __init__(self, fileName=''):
Thread.__init__(self)
super(Temp, self).__init__()
self._stop = threading.Event()
self.tempDir = '/sys/bus/w1/devices/'
list = os.listdir(self.tempDir)
if(list[0][:2]=="28"):
fileName=list[0]
self.fileName = fileName
self.currentTemp = -999
self.correctionFactor = 1;
self.enabled = True
self.Run=True
def run(self):
while self.isEnabled():
try:
f = open(self.tempDir + self.fileName + "/w1_slave", 'r')
except IOError as e:
print "Error: File " + self.tempDir + self.fileName + "/w1_slave" + " does not exits"
return;
lines=f.readlines()
crcLine=lines[0]
tempLine=lines[1]
result_list = tempLine.split("=")
temp = float(result_list[-1])/1000 # temp in Celcius
temp = temp + self.correctionFactor # correction factor
#if you want to convert to Celcius, comment this line
temp = (9.0/5.0)*temp + 32
if crcLine.find("NO") > -1:
temp = -999
self.currentTemp = temp
#print "Current: " + str(self.currentTemp) + " " + str(self.fileName)
sleep(.5)
#returns the current temp for the probe
def getCurrentTemp(self):
return self.currentTemp
#setter to enable this probe
def setEnabled(self, enabled):
self.enabled = enabled
#getter
def isEnabled(self):
return self.enabled
def updating_writer(a):
context = a[0]
register = 3
slave_id = 0x00
address = 0x00
global pwmDutyCycle,temp
#uncomment to debug temperature
print temp.getCurrentTemp()
values = [int(pwmDutyCycle),temp.getCurrentTemp()*100]
context[slave_id].setValues(register,address,values)
def read_context(a):
context = a[0]
register = 3
slave_id = 0x00
address = 0x00
value = context[slave_id].getValues(register,address)[0]
global pwmDutyCycle
if(value!=pwmDutyCycle):
print value
pwmDutyCycle=value
pwm.ChangeDutyCycle(pwmDutyCycle)
def main():
store = ModbusSlaveContext(
di = ModbusSequentialDataBlock(0, [0]*100),
co = ModbusSequentialDataBlock(0, [0]*100),
hr = ModbusSequentialDataBlock(0, [0]*100),
ir = ModbusSequentialDataBlock(0, [0]*100))
context = ModbusServerContext(slaves=store, single=True)
identity = ModbusDeviceIdentification()
identity.VendorName = 'pymodbus'
identity.ProductCode = 'PM'
identity.VendorUrl = 'http://github.com/simplyautomationized'
identity.ProductName = 'pymodbus Server'
identity.ModelName = 'pymodbus Server'
identity.MajorMinorRevision = '1.0'
time = 5 # 5 seconds delaytime = 5 # 5 seconds delay
writer = LoopingCall(read_context,a=(context,))
loop = LoopingCall(updating_writer, a=(context,))
loop.start(.5) # initially delay by time
writer.start(.1)
StartTcpServer(context, identity=identity)#, address=("localhost", 502))
#cleanup async tasks
temp.setEnabled(False)
loop.stop()
writer.stop()
GPIO.cleanup()
if __name__ == "__main__":
temp = Temp()
temp.start()
main()
This comment has been removed by the author.
ReplyDeleteHello I watched your video Raspberry Pi as ModbusTCP Slave but i have some question, i need to read some data sensor's in the raspberry py and sending in a static ip node in a server, also i need to read the data sending from the Raspberry Pi with top server on a pc .
ReplyDeleteCuold you explain me how it pyModbus library works? and if you help me with my problem I appreciate it
I have the part of reading data sensor's but i need the part of transfer data to a server
Delete