mirror of https://github.com/lumapu/ahoy.git
Martin Grill
3 years ago
5 changed files with 168 additions and 10 deletions
@ -0,0 +1,45 @@ |
|||||
|
#include "common.hpp" |
||||
|
|
||||
|
#include <sstream> |
||||
|
#include <iostream> |
||||
|
#include <iomanip> |
||||
|
|
||||
|
using namespace std; |
||||
|
|
||||
|
/** Convert given 5-byte address to human readable hex string */ |
||||
|
string prettyPrintAddr(string &a) |
||||
|
{ |
||||
|
ostringstream o; |
||||
|
o << hex << setw(2) |
||||
|
<< setfill('0') << setw(2) << int(a[0]) |
||||
|
<< ":" << setw(2) << int(a[1]) |
||||
|
<< ":" << setw(2) << int(a[2]) |
||||
|
<< ":" << setw(2) << int(a[3]) |
||||
|
<< ":" << setw(2) << int(a[4]) << dec; |
||||
|
return o.str(); |
||||
|
} |
||||
|
|
||||
|
|
||||
|
/** Convert a Hoymiles inverter/DTU serial number into its
|
||||
|
* corresponding NRF24 address byte sequence (5 bytes). |
||||
|
* |
||||
|
* The inverters use a BCD representation of the last 8 |
||||
|
* digits of their serial number, in reverse byte order, |
||||
|
* followed by \x01. |
||||
|
*/ |
||||
|
string serno2shockburstaddrbytes(uint64_t n) |
||||
|
{ |
||||
|
char b[5]; |
||||
|
b[3] = (((n/10)%10) << 4) | ((n/1)%10); |
||||
|
b[2] = (((n/1000)%10) << 4) | ((n/100)%10); |
||||
|
b[1] = (((n/100000)%10) << 4) | ((n/10000)%10); |
||||
|
b[0] = (((n/10000000)%10) << 4) | ((n/1000000)%10); |
||||
|
b[4] = 0x01; |
||||
|
|
||||
|
string s = string(b, sizeof(b)); |
||||
|
|
||||
|
cout << dec << "ser# " << n << " --> addr " |
||||
|
<< prettyPrintAddr(s) << endl; |
||||
|
return s; |
||||
|
} |
||||
|
|
@ -0,0 +1,18 @@ |
|||||
|
#include <string> |
||||
|
#include <stdint.h> |
||||
|
|
||||
|
using namespace std; |
||||
|
|
||||
|
/** Convert given 5-byte address to human readable hex string */ |
||||
|
string prettyPrintAddr(string &a); |
||||
|
|
||||
|
|
||||
|
/** Convert a Hoymiles inverter/DTU serial number into its
|
||||
|
* corresponding NRF24 address byte sequence (5 bytes). |
||||
|
* |
||||
|
* The inverters use a BCD representation of the last 8 |
||||
|
* digits of their serial number, in reverse byte order, |
||||
|
* followed by \x01. |
||||
|
*/ |
||||
|
string serno2shockburstaddrbytes(uint64_t n); |
||||
|
|
@ -0,0 +1,90 @@ |
|||||
|
/* based on "gettingstarted.cpp" by 2bdy5 */ |
||||
|
|
||||
|
/**
|
||||
|
* Behave like we expect a Hoymiles microinverter to behave. |
||||
|
*/ |
||||
|
#include <ctime> // time() |
||||
|
#include <iostream> // cin, cout, endl |
||||
|
#include <iomanip> |
||||
|
#include <string> // string, getline() |
||||
|
#include <vector> |
||||
|
#include <sstream> |
||||
|
#include <time.h> // CLOCK_MONOTONIC_RAW, timespec, clock_gettime() |
||||
|
#include <RF24/RF24.h> // RF24, RF24_PA_LOW, delay() |
||||
|
|
||||
|
using namespace std; |
||||
|
|
||||
|
#include "common.hpp" |
||||
|
|
||||
|
// Generic:
|
||||
|
RF24 radio(22, 0); |
||||
|
// See http://nRF24.github.io/RF24/pages.html for more information on usage
|
||||
|
|
||||
|
|
||||
|
/** Receive forever
|
||||
|
*/ |
||||
|
void receiveForever(int ch, string myaddr) |
||||
|
{ |
||||
|
uint8_t buf[30]; |
||||
|
|
||||
|
radio.enableDynamicPayloads(); |
||||
|
radio.setChannel(ch); |
||||
|
|
||||
|
radio.setPALevel(RF24_PA_MIN); // RF24_PA_MAX is default.
|
||||
|
radio.setDataRate(RF24_250KBPS); |
||||
|
|
||||
|
// set the RX address of the TX node into a RX pipe
|
||||
|
radio.openReadingPipe(1, (const uint8_t *)myaddr.c_str()); |
||||
|
|
||||
|
while (true) |
||||
|
{ |
||||
|
uint8_t pipe; |
||||
|
if (radio.available(&pipe)) |
||||
|
{ |
||||
|
uint8_t bytes = radio.getPayloadSize(); // get the size of the payload
|
||||
|
cout << "I was notified of having received " << (unsigned int)bytes; |
||||
|
cout << " bytes on pipe " << (unsigned int)pipe << flush; |
||||
|
radio.read(buf, bytes); // fetch payload from FIFO
|
||||
|
//cout << ": " << payload; // print the payload's value
|
||||
|
//cout << " hex: " << hex << (unsigned int)b[0] << " " << (unsigned int)b[1] << " "
|
||||
|
// << (unsigned int)b[2] << " " << (unsigned int)b[3] << " " <<endl;
|
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
|
||||
|
int main(int argc, char** argv) |
||||
|
{ |
||||
|
if (!radio.begin()) { |
||||
|
cout << "radio hardware is not responding!!" << endl; |
||||
|
return 0; // quit now
|
||||
|
} |
||||
|
|
||||
|
if(!radio.isPVariant()) |
||||
|
{ |
||||
|
printf("not nRF24L01+\n"); |
||||
|
return 0; |
||||
|
} |
||||
|
|
||||
|
if(!radio.isChipConnected()) |
||||
|
{ |
||||
|
printf("not connected\n"); |
||||
|
return 0; |
||||
|
} |
||||
|
|
||||
|
// TODO
|
||||
|
// we probably want
|
||||
|
// - 8-bit crc
|
||||
|
// - dynamic payloads (check in rf logs)
|
||||
|
// - what's the "primary mode"?
|
||||
|
// - do we need/want "custom ack payloads"?
|
||||
|
// - use isAckPayloadAvailable() once we've actually contacted an inverter successfully!
|
||||
|
|
||||
|
radio.printPrettyDetails(); |
||||
|
|
||||
|
string addr = serno2shockburstaddrbytes(114174608177); |
||||
|
receiveForever(41, addr); |
||||
|
|
||||
|
return 0; |
||||
|
} |
||||
|
|
Loading…
Reference in new issue