Propagation paths from WSPRnet |
WSPR is unique that there are several automated receiving stations which reports the signals and shows it on a world map so that you can easily see the propagation patterns (see here). Each WSPR transmission will start at the top on the even minutes in UTC. So exact time synchronization is a must. Secondly the signals are encoded in frequency shifted keying (hence needs a stable oscillator)
Amazed by the potentials of whisper and the lack of space to put up a proper antenna made me to give a try with this protocol. So i decided to test it out with an AD9850 ebay module (dds signal generator) and an arduino uno. To boost up the output from the module a little bit, i hooked up a small power-amp from a frog sound qrp kit (popular on ebay/aliexpress)with a low pass filter ( for testing i simple removed NE602 from the socket and connected the AD9850 output to pin 3 on ic socket & ground to dds ground, also make sure that you use the Q1 9018 and feed 9 volt to keep the power output low and to keep the d882 safe from damage ). Hooked up a random length of wire (i got from a cat 5 cable, about 5 mtr long) to the antenna socket and a similar piece as counterpoise). In this testing , i put the wire on the curtain of the room and the counter poise on the floor! so i had the worst possible antenna and swr )
Simple direct conversion cw trx |
After the test i checked on the WSPRnet website and was amazed to see the spots a 1000km away!!. This was done for testing and a solution with gps sync and a proper antenna & swr matching will do the best for long run. It is also important that the tx should be on the standard whisper frequencies (see here). As my LPF was for 40 mtr , i did my test on 7mhz band. But most popular one for wspr is 30mtr band.
For a constant use , i would recommend the kit from hansummers which costs a lot less for the entire setup (See: http://www.hanssummers.com/ultimate3.html)
Results
Got a couple of spots from dx stations located several miles away. Snr was not so good, but as a proof of concept , the bad antenna and quick set up with manual utc trigger worked!
Spots on 40mtrs |
and finally here is my messy setup followed by the arduino code (courtesy to http://www.george-smart.co.uk/wiki/Arduino_WSPR).
wspr testing with ad9850 and arduino uno |
- More on ad 9850 module connection can be read here and datasheet
- For generating the tone array for the sketch , i used genwspr.exe listed at http://www.g4jnt.com/jtmodesbcns.htm. You get a list of numbers and use search replace function to put a set of commas in between the numbers and paste it to arduino ide. If you need the source/logic for making this by yourself you can get them (as well as the exe) from here
/*
* A simple single freq AD9850 Arduino test script
* Original AD9851 DDS sketch by Andrew Smallbone at www.rocketnumbernine.com
* Modified for testing the inexpensive AD9850 ebay DDS modules
* Pictures and pinouts at nr8o.dhlpilotcentral.com
* 9850 datasheet at http://www.analog.com/static/imported-files/data_sheets/AD9850.pdf
* From http://www.george-smart.co.uk/wiki/Arduino_WSPR
* modification by http://blog.riyas.org
* Use freely
*/
#define W_CLK 8 // Pin 8 - connect to AD9850 module word load clock pin (CLK)
#define FQ_UD 9 // Pin 9 - connect to freq update pin (FQ)
#define DATA 10 // Pin 10 - connect to serial data load pin (DATA)
#define RESET 11 // Pin 11 - connect to reset pin (RST).
#define LED 13 //flashes duty cycle
#define SWITCH A0 // connect this to a switch and the other end to vcc via 10k resistor
#define CLOCKLED 3 //flash every seconds
#define pulseHigh(pin) {digitalWrite(pin, HIGH); digitalWrite(pin, LOW); }
#define WSPR_TX_B 7.040000e6 // this is the bottom of the band. The station moves about.
// DDS Offset in Hz
#define DDS_OSET 50 //DDS #2
#define WSPR_DUTY 3 // transmit every N slices.
// WSPR Tone Data - line breaks in no parciular place, just to look pretty. (0-161).
//callsign locator power in dbm
// to bulid this array see:http://www.george-smart.co.uk/wiki/Arduino_WSPR
static byte WSPR_DATA_HOME[] = {1,1,0,0,1,1,0,0,3,0,0,2,3,3,1,2,2,2,1,2,2,1,0,3,3,3,3,0,0,2,0,2,
...................................,2};
#define WSPR_DATA WSPR_DATA_HOME
#include <stdint.h>
// Variables
unsigned long WSPR_TXF = 0;
unsigned long minute =1;
unsigned long seconds=0;
unsigned long tstart=0;
unsigned long startT = 0, stopT = 0;
char sz[32];
int duty=0;
int val=0;
int band=0;
// transfers a byte, a bit at a time, LSB first to the 9850 via serial DATA line
void tfr_byte(byte data)
{
for (int i=0; i<8; i++, data>>=1) {
digitalWrite(DATA, data & 0x01);
pulseHigh(W_CLK); //after each bit sent, CLK is pulsed high
}
}
// frequency calc from datasheet page 8 = <sys clock> * <frequency tuning word>/2^32
void sendFrequency(double frequency) {
int32_t freq = frequency * 4294967295/125000000; // note 125 MHz clock on 9850
for (int b=0; b<4; b++, freq>>=8) {
tfr_byte(freq & 0xFF);
}
tfr_byte(0x000); // Final control byte, all 0 for 9850 chip
pulseHigh(FQ_UD); // Done! Should see output
}
void setup() {
// configure arduino data pins for output
pinMode(FQ_UD, OUTPUT);
pinMode(W_CLK, OUTPUT);
pinMode(DATA, OUTPUT);
pinMode(RESET, OUTPUT);
pulseHigh(RESET);
pulseHigh(W_CLK);
pulseHigh(FQ_UD); // this pulse enables serial mode - Datasheet page 12 figure 10
pinMode (LED, OUTPUT);
pinMode (CLOCKLED, OUTPUT);
pinMode(SWITCH, INPUT);
Serial.begin(9600);
sprintf(sz, "\nLB7UG Compiled %s %s", __TIME__, __DATE__);
Serial.println(sz);
Serial.print("\n\nDDS Reset ");
delay(900);
delay(100);
Serial.println("OK");
duty = 0;
}
void loop() {
//let us make a poor mans clock which needs a click on SWITCH pin to high at an even minute
val=analogRead(SWITCH);
//Serial.println(val);
if (val == 1023) {
//reset the clock here
tstart = millis()/1000UL;
wsprTX();
Serial.println(tstart);
}
//WSPR_TXF = (WSPR_TX_B+DDS_OSET) + random(10, 190); // always choose a frequency, a little with the pRNG.
WSPR_TXF = (WSPR_TX_B+DDS_OSET); // always choose a frequency, a little with the pRNG.
//wsprTX();
// UPDATING TIME DO NOT INSERT COMMANDS BELOW!!
seconds=(millis()/1000UL)-tstart;
minute= seconds/60UL;
//Serial.println(minute);
if ( (minute % 2 == 0) && (minute > 1) && (seconds >= 1) && (seconds <= 4) ) { // start transmission
//if (duty % WSPR_DUTY == 0) {
if (1) {
//Serial.print("Beginning WSPR Transmission on ");
//Serial.print(WSPR_TXF-DDS_OSET);
//Serial.println(" Hz.");
wsprTX();
duty++;
band++;
//Serial.println(" Transmission Finished.");
} else {
duty++;
digitalWrite (LED, LOW);
flash_led(WSPR_DUTY, LED); // flash the WSPR duty.
}
}
//update time again
seconds=(millis()/1000UL)-tstart;
if ((seconds % 2 ==0) && (tstart!=0)){ // oru bhangikku vendi
digitalWrite(CLOCKLED, HIGH);
}
else
{
digitalWrite(CLOCKLED, LOW);
}
}
void flash_led(unsigned int t, int l) {
unsigned int i = 0;
if (t > 25) {
digitalWrite(l, HIGH);
delay(2000);
digitalWrite(l, LOW);
} else {
for (i=0;i<t;i++) {
digitalWrite(l, HIGH);
delay(250);
digitalWrite(l, LOW);
delay(250);
}
}
}
void wsprTX() {
int i = 0;
digitalWrite(LED, HIGH);
for (i=0;i<162;i++) {
wsprTXtone( WSPR_DATA[i] );
delay(682);
}
sendFrequency(0);
digitalWrite(LED, LOW);
}
void wsprTXtone(int t) {
if ((t >= 0) && (t <= 3) ) {
//Serial.print((WSPR_TXF + (t * 2)));
sendFrequency((WSPR_TXF + (t * 2))); // should really be 1.4648 Hz not 2.
} else {
Serial.print("Tone #");
Serial.print(t);
Serial.println(" is not valid. (0 <= t <= 3).");
}
}
end
I've been thinking along the same lines, only reason its not happening yet is time! CT7AOX, ex G8CYE. 73
ReplyDelete