The RMS RPI MOD |
||
- Description
- Parts List
- RMS to DC Convertion
- 12-bit Analog to Digital Convertion
- Photos
- Software
- Contact Us
Language
Note this software is written for the 30Amps/Volt YHDC split-core transformer. [YHDC Data Sheet]
C++ Software
First install wiringPi.
wiringPi
Command Lines
g++-4.7 -std=c++11 -c -I/usr/local/include -o amps.o amps.c++ g++-4.7 -std=c++11 -I/usr/local/include -o amps amps.o -L/usr/local/lib -lwiringPiDownload Header
Download Source
Header
/*
* amps.h
*
* Acme Software Works
*/
/*************************************************************/
/*
* Example code for the Acme Software Works RMS RPI MOD
*/
/*************************************************************/
#ifndef _amps_h_
#define _amps_h_
#define OPTIONS "h?dc:"
#define USAGE "Usage: %s [-d (debug)] [-c 1|2 (channel)]\n"
/*************************************************************/
using namespace std;
/*************************************************************/
/* Classes */
class Amps {
int fd;
bool debug;
char time_str[26];
int channel;
double amps;
public:
Amps(int channel, bool debug = false) { init(channel,debug); };
enum { CHANNEL1 = 0x4d, CHANNEL2 = 0x4e };
void init(int channel, bool debug = false);
double readAmps(void);
const char* getTimeStr(void);
void printAmps() {
cout << getTimeStr() << " amps: " << setprecision(2) << amps << '\n';
};
};
/*************************************************************/
#endif /* _amps_h_ */
Implementation
/*
* amps.c++
*
* Acme Software Works
*
*/
/*************************************************************/
/*
*
* Example code for the Acme Software Works RMS RPI MOD
*
*/
/*************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <string>
#include <inttypes.h>
#include <iostream>
#include <iomanip>
#include <wiringPi.h>
#include <wiringPiI2C.h>
/*************************************************************/
#include "amps.h"
/*************************************************************/
/*
* Needed for command line parsing (getopt())
*/
extern char* optarg;
extern int optind;
char* my_name;
/*************************************************************/
/*
* main
*
*/
main(int argc,char* argv[])
{
int c;
bool debug = false;
int channel = Amps::CHANNEL1;
/* Get command line opts */
if((my_name = strrchr(argv[0],'/')) == NULL) {
my_name = argv[0];
} else ++my_name;
while((c = getopt(argc,argv,OPTIONS)) != -1) {
switch(c) {
case 'c' :
{
int ch = atoi(optarg);
if ( ch == 1 ) channel = Amps::CHANNEL1;
else if ( ch == 2 ) channel = Amps::CHANNEL2;
else {
fprintf(stderr,"No such channel\n");
fprintf(stderr,USAGE,my_name);
exit(1);
}
}
break;
case 'd' :
debug = true;
break;
case '?' :
fprintf(stderr,USAGE,my_name);
exit(1);
}
}
argc -= optind;
argv += optind;
{
Amps monitor(channel,debug);
for(;;) {
sleep(1);
monitor.readAmps();
monitor.printAmps();
}
}
exit(0);
} /* End of main() */
/*************************************************************/
const char* Amps::getTimeStr()
{
time_t result = time(NULL);
strftime(time_str,sizeof(time_str),"%FT%T",localtime(&result));
return time_str;
} // End of getTimeStr()
/*************************************************************/
double Amps::readAmps()
{
uint8_t buf[3];
int16_t val;
double rms_val = 0.0;
wiringPiI2CWrite(fd,0xab);
if (read(fd, buf, 2) != 2) {
perror("Read conversion");
return -1.0;
}
val = (int16_t)buf[0]*256 + (uint16_t)buf[1];
rms_val = (double)val*(5.0/(double)4096);
if ( debug )
cout << getTimeStr() << " volts: " << setprecision(4) << rms_val << '\n';
amps = 30.0*rms_val; // 30 amps per volt split-core transformer
return amps;
} // End of readAmps()
/*************************************************************/
void Amps::init(int channel,bool debug)
{
this->debug = debug;
this->channel = channel;
if (wiringPiSetup() == -1) {
cout << "Setup failed" << '\n';
exit(1);
}
// MCP3221 A5 | A6
if ((fd = wiringPiI2CSetup(channel)) == -1) {
cout << "I2C Setup failed" << '\n';
exit(1);
}
} // End of init()
/*************************************************************/
/******************** END OF FILE ****************************/
/*************************************************************/
C Software
First install wiringPi.
wiringPi
Command Lines
gcc-4.7 -std=c11 -c -I/usr/local/include -o amps.o amps.c
gcc-4.7 -std=c11 -I/usr/local/include -o amps amps.o -L/usr/local/lib -lwiringPi
Download
/*
* amps.c
*
* Acme Software Works
*
*/
/*************************************************************/
/*
*
* Example code for the Acme Software Works RMS RPI MOD
*
*/
/*************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
#include <string.h>
#include <unistd.h>
#include <time.h>
#include <inttypes.h>
#include <wiringPi.h>
#include <wiringPiI2C.h>
/*************************************************************/
// Function Definitions
const char* getTimeStr();
double readAmps(int fd,int debug);
int init(int channel);
/*************************************************************/
/*
* Needed for command line parsing (getopt())
*/
int getopt(int argc, char * const argv[], const char *optstring);
extern char* optarg;
extern int optind;
char* my_name;
/*************************************************************/
/*
* main
*
*/
int main(int argc,char* argv[])
{
const char *USAGE = "Usage: %s [-d (debug)] [-c 1|2 (channel)]\n";
// MCP3221 I2C addresses
int const CHANNEL1 = 0x4d;
int const CHANNEL2 = 0x4e;
int c;
int debug = FALSE;
int channel = CHANNEL1;
int fd = -1;
/* Get command line opts */
if((my_name = strrchr(argv[0],'/')) == NULL) {
my_name = argv[0];
} else ++my_name;
while((c = getopt(argc,argv,"h?dc:")) != -1) {
switch(c) {
case 'c' :
{
int ch = atoi(optarg);
if ( ch == 1 ) channel = CHANNEL1;
else if ( ch == 2 ) channel = CHANNEL2;
else {
fprintf(stderr,"No such channel\n");
fprintf(stderr,USAGE,my_name);
exit(1);
}
}
break;
case 'd' :
debug = TRUE;
break;
case '?' :
fprintf(stderr,USAGE,my_name);
exit(1);
}
}
argc -= optind;
argv += optind;
{
fd = init(channel);
for(;;) {
sleep(1);
printf("%s amps: %.2lf\n",getTimeStr(),readAmps(fd,debug));
}
}
exit(0);
} /* End of main() */
/*************************************************************/
const char* getTimeStr()
{
static char time_str[26] = "";
time_t result = time(NULL);
strftime(time_str,sizeof(time_str),"%FT%T",localtime(&result));
return time_str;
} // End of getTimeStr()
/*************************************************************/
double readAmps(int fd,int debug)
{
uint8_t buf[3];
int16_t val;
double rms_val = 0.0;
double amps = 0.0;
wiringPiI2CWrite(fd,0xab);
if (read(fd, buf, 2) != 2) {
perror("Read conversion");
return -1.0;
}
val = (int16_t)buf[0]*256 + (uint16_t)buf[1];
rms_val = (double)val*(5.0/(double)4096);
if ( debug ) printf("%s volts: %.4lf\n",getTimeStr(),rms_val);
amps = 30.0*rms_val; // 30 amps per volt split-core transformer
return amps;
} // End of readAmps()
/*************************************************************/
int init(int channel)
{
int fd = -1;
if (wiringPiSetup() == -1) {
printf("Setup failed\n") ;
exit(1);
}
// MCP3221 A5 | A6
if ((fd = wiringPiI2CSetup(channel)) == -1) {
printf("I2C Setup failed\n") ;
exit(1);
}
return fd;
} // End of init()
/*************************************************************/
/******************** END OF FILE ****************************/
/*************************************************************/
Python Software
First install python-smbus.
python-smbus
Download
#! /usr/bin/python
##########################################################
#
# Acme Software Works
#
##########################################################
##########################################################
#
# Example code for the Acme Software Works RMS RPI MOD
#
##########################################################
import smbus
import time
from time import gmtime, strftime
import datetime
import getopt, sys
CHANNEL1 = 0x4d
CHANNEL2 = 0x4e
def usage():
print "Usage: amps.py [-d (debug)] [-c 1|2 (channel)]\n"
def readAmps(ch,debug):
bus = smbus.SMBus(1)
if ch == "2":
addr = CHANNEL2
else:
addr = CHANNEL1
data = bus.read_i2c_block_data(addr, 1,2)
val = (data[0] << 8) + data[1]
rms_val = float(val)*(5.0/float(4096));
amps = 30.0*rms_val # 30 amps per volt split-core transformer
now = strftime("%Y-%m-%d %H:%M:%S", gmtime())
if debug == True:
print now,"volts:",'{:.4f}'.format(rms_val)
print now," amps:",'{:.2f}'.format(amps)
time.sleep(1)
# main
try:
opts, args = getopt.getopt(sys.argv[1:], "dc:")
except getopt.GetoptError as err:
print str(err)
usage()
sys.exit(2)
ch = "1"
debug = False
for o, a in opts:
if o == "-d":
debug = True
elif o == "-c":
if a in ("1","2"):
ch = a
else:
usage()
sys.exit(2)
else:
assert False, "unhandled option"
while True:
readAmps(ch,debug)
##### End of File ######
For more information: info@acmesoftwareworks.com