You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
395 lines
11 KiB
395 lines
11 KiB
// Sends CubeSatSim telemetry encoded as CW (Morse Code) using AO-7 format
|
|
//
|
|
// Portions Copyright (c) 2018 Brandenburg Tech, LLC
|
|
// All right reserved.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY BRANDENBURG TECH, LLC AND CONTRIBUTORS
|
|
// ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
// PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BRANDENBURT TECH, LLC
|
|
// AND CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
// OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
// WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
// OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
// ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
#include <axradio/axradioinit_p.h>
|
|
#include <axradio/axradiomode_p.h>
|
|
#include <axradio/axradiorx_p.h>
|
|
#include <axradio/axradiotx_p.h>
|
|
#include <generated/configtx.h>
|
|
#include <pthread.h>
|
|
#include <semaphore.h>
|
|
#include <spi/ax5043spi_p.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
|
|
#define MAX_MESSAGE_LENGTH (197)
|
|
|
|
extern uint8_t axradio_rxbuffer[];
|
|
void *transmit(void *arg);
|
|
int get_message(uint8_t *buffer, int avail);
|
|
int lower_digit(int number);
|
|
int upper_digit(int number);
|
|
int encode_digit(uint8_t *msg, int number);
|
|
void config_cw();
|
|
int encode_tlm(uint8_t *buffer, int channel, int val1, int val2, int val3, int val4, int avail);
|
|
int encode_header(uint8_t *buffer, int avail);
|
|
int add_dash(uint8_t *msg, int number);
|
|
int add_dot(uint8_t *msg, int number);
|
|
int add_space(uint8_t *msg);
|
|
|
|
int main(void)
|
|
{
|
|
uint8_t retVal;
|
|
|
|
// Configure SPI bus to AX5043
|
|
setSpiChannel(SPI_CHANNEL);
|
|
setSpiSpeed(SPI_SPEED);
|
|
initializeSpi();
|
|
|
|
// Initialize the AX5043
|
|
retVal = axradio_init();
|
|
if (retVal == AXRADIO_ERR_NOCHIP) {
|
|
fprintf(stderr, "ERROR: No AX5043 RF chip found\n");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
if (retVal != AXRADIO_ERR_NOERROR) {
|
|
fprintf(stderr, "ERROR: Unable to initialize AX5043\n");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
printf("INFO: Found and initialized AX5043\n");
|
|
|
|
retVal = mode_tx();
|
|
if (retVal != AXRADIO_ERR_NOERROR) {
|
|
fprintf(stderr, "ERROR: Unable to enter TX mode\n");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
config_cw();
|
|
|
|
// allocate space for the buffer
|
|
static uint8_t packet[MAX_MESSAGE_LENGTH + 1];
|
|
|
|
int channel; // AO-7 telemetry format has 6 channels, 4 sub channels in each
|
|
int msg_length;
|
|
|
|
FILE* file = popen("mpcmd show data 2>&1", "r");
|
|
char cmdbuffer[1000];
|
|
fgets(cmdbuffer, 1000, file);
|
|
pclose(file);
|
|
printf("buffer is :%s\n", cmdbuffer);
|
|
|
|
char mopower[64][14];
|
|
|
|
|
|
// char str[] ="- This, a sample string.";
|
|
char * pch;
|
|
// printf ("Splitting string \"%s\" into tokens:\n",str);
|
|
// pch = strtok (str," ");
|
|
int i = 0;
|
|
pch = strtok (cmdbuffer," ,.-");
|
|
while (pch != NULL)
|
|
{
|
|
strcpy(mopower[i], pch);
|
|
printf ("mopwer[%d]=%s\n",i,mopower[i]); // pch);
|
|
pch = strtok (NULL, " ");
|
|
i++;
|
|
}
|
|
printf("Battery voltage = %s\n", mopower[16]);
|
|
|
|
int tlm_3a = (int)(((strtol(mopower[16], (char **)NULL, 10) * 10) - 65) + 0.5);
|
|
|
|
printf("TLM 3A = %d \n", tlm_3a);
|
|
|
|
while(1) { // loop forever
|
|
|
|
for (channel = 0; channel < 7; channel++) {
|
|
|
|
if (channel == 0) { // start with telemetry header "hi hi" plus a few chars to help CW decoding software sync
|
|
msg_length = encode_header(&packet[0], MAX_MESSAGE_LENGTH + 1);
|
|
|
|
printf("\nINFO: Sending TLM header\n");
|
|
|
|
} else {
|
|
msg_length = encode_tlm(&packet[0], channel, // add a channel with dummy data to buffer
|
|
channel, channel+1, channel+2, channel+3,
|
|
(MAX_MESSAGE_LENGTH + 1));
|
|
|
|
printf("\nINFO: Sending TLM channel %d \n", channel);
|
|
}
|
|
printf("DEBUG: msg_length = %d\n", msg_length);
|
|
|
|
retVal = transmit_packet(&remoteaddr_tx, packet, (uint16_t)(msg_length)); // send telemetry
|
|
if (retVal != AXRADIO_ERR_NOERROR) {
|
|
fprintf(stderr, "ERROR: Unable to transmit a packet\n");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
// sleep(1);
|
|
|
|
usleep(200000);
|
|
}
|
|
}
|
|
}
|
|
// Encodes telemetry header (channel 0) into buffer
|
|
//
|
|
int encode_header(uint8_t *buffer, int avail) {
|
|
|
|
int count = 0;
|
|
count += add_space(&buffer[count]);
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_dash(&buffer[count], 1); // c
|
|
count += add_dot(&buffer[count], 1);
|
|
count += add_dash(&buffer[count], 1);
|
|
count += add_dot(&buffer[count], 1);
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_dash(&buffer[count], 2); // q
|
|
count += add_dot(&buffer[count], 1);
|
|
count += add_dash(&buffer[count], 1);
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_dot(&buffer[count], 4); // h
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_dot(&buffer[count], 2); // i
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_dot(&buffer[count], 4); // h
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_dot(&buffer[count], 2); // i
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += add_space(&buffer[count]);
|
|
count += add_space(&buffer[count]);
|
|
|
|
return count;
|
|
}
|
|
|
|
// Encodes one channel of telemetry into buffer
|
|
//
|
|
int encode_tlm(uint8_t *buffer, int channel, int val1, int val2, int val3, int val4, int avail) {
|
|
|
|
int count = 0;
|
|
|
|
count += add_space(&buffer[count]);
|
|
count += add_space(&buffer[count]);
|
|
count += add_space(&buffer[count]);
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += encode_digit(&buffer[count], channel); // for channel 1, encodes 1aa
|
|
count += encode_digit(&buffer[count], upper_digit(val1));
|
|
count += encode_digit(&buffer[count], lower_digit(val1));
|
|
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += encode_digit(&buffer[count], channel); // for channel 1, encodes 1bb
|
|
count += encode_digit(&buffer[count], upper_digit(val2));
|
|
count += encode_digit(&buffer[count], lower_digit(val2));
|
|
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += encode_digit(&buffer[count], channel); // for channel 1, encodes 1cc
|
|
count += encode_digit(&buffer[count], upper_digit(val3));
|
|
count += encode_digit(&buffer[count], lower_digit(val3));
|
|
|
|
count += add_space(&buffer[count]);
|
|
|
|
count += encode_digit(&buffer[count], channel); // for channel 1, encodes 1dd
|
|
count += encode_digit(&buffer[count], upper_digit(val4));
|
|
count += encode_digit(&buffer[count], lower_digit(val4));
|
|
|
|
count += add_space(&buffer[count]);
|
|
|
|
//printf("DEBUG count: %d avail: %d \n", count, avail);
|
|
if (count > avail) { // make sure not too long
|
|
buffer[avail-1] = 0;
|
|
count = avail-1;
|
|
printf("DEBUG count > avail!\n");
|
|
}
|
|
return count;
|
|
}
|
|
// Encodes a single digit of telemetry into buffer
|
|
//
|
|
int encode_digit(uint8_t *buffer, int digit) {
|
|
int count = 0;
|
|
switch(digit)
|
|
{
|
|
case 0:
|
|
count += add_dash(&buffer[count], 5); // 0
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 1:
|
|
count += add_dot(&buffer[count], 1); // 1
|
|
count += add_dash(&buffer[count], 4);
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 2:
|
|
count += add_dot(&buffer[count], 2); // 2
|
|
count += add_dash(&buffer[count], 3);
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 3:
|
|
count += add_dot(&buffer[count], 3); // 3
|
|
count += add_dash(&buffer[count], 2);
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 4:
|
|
count += add_dot(&buffer[count], 4); // 4
|
|
count += add_dash(&buffer[count], 1);
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 5:
|
|
count += add_dot(&buffer[count], 5); // 5
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 6:
|
|
count += add_dash(&buffer[count], 1); // 6
|
|
count += add_dot(&buffer[count], 4);
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 7:
|
|
|
|
count += add_dash(&buffer[count], 2); // 7
|
|
count += add_dot(&buffer[count], 3);
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 8:
|
|
count += add_dash(&buffer[count], 3); // 8
|
|
count += add_dot(&buffer[count], 2);
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
case 9:
|
|
count += add_dash(&buffer[count], 4); // 9
|
|
count += add_dot(&buffer[count], 1);
|
|
count += add_space(&buffer[count]);
|
|
|
|
break;
|
|
default:
|
|
printf("ERROR: Not a digit!\n");
|
|
return 0;
|
|
}
|
|
return count;
|
|
}
|
|
// Returns lower digit of a number which must be less than 99
|
|
//
|
|
int lower_digit(int number) {
|
|
|
|
int digit = 0;
|
|
|
|
if (number < 100)
|
|
digit = number - ((int)(number/10) * 10);
|
|
else
|
|
printf("ERROR: Not a digit in lower_digit!\n");
|
|
|
|
return digit;
|
|
}
|
|
// Returns upper digit of a number which must be less than 99
|
|
//
|
|
int upper_digit(int number) {
|
|
|
|
int digit = 0;
|
|
|
|
if (number < 100)
|
|
digit = (int)(number/10);
|
|
else
|
|
printf("ERROR: Not a digit in upper_digit!\n");
|
|
|
|
return digit;
|
|
}
|
|
// Configure radio to send CW which is ASK
|
|
//
|
|
void config_cw() {
|
|
|
|
printf("Register write to clear framing and crc\n");
|
|
ax5043WriteReg(0x12,0);
|
|
|
|
printf("Register write to disable fec\n");
|
|
ax5043WriteReg(0x18,0);
|
|
|
|
printf("Register write \n");
|
|
ax5043WriteReg(0x165,0);
|
|
|
|
ax5043WriteReg(0x166,0);
|
|
ax5043WriteReg(0x167,0x50); // 0x25); // 0x50); // 0x08); // 0x20);
|
|
|
|
ax5043WriteReg(0x161,0);
|
|
ax5043WriteReg(0x162,0x20);
|
|
|
|
long txRate;
|
|
txRate = ax5043ReadReg(0x167) + 256 * ax5043ReadReg(0x166) + 65536 * ax5043ReadReg(0x165);
|
|
printf("Tx Rate %x %x %x \n", ax5043ReadReg(0x165), ax5043ReadReg(0x166), ax5043ReadReg(0x167));
|
|
long fskDev;
|
|
fskDev = ax5043ReadReg(0x163) + 256 * ax5043ReadReg(0x162) + 65536 * ax5043ReadReg(0x161);
|
|
|
|
ax5043WriteReg(0x37,(uint8_t)((ax5043ReadReg(0x37) + 4))); // Increase FREQA
|
|
|
|
printf("Tx Rate: %ld FSK Dev: %ld \n", txRate, fskDev);
|
|
|
|
ax5043WriteReg(0x10,0); // ASK
|
|
|
|
printf("Modulation: %x \n", (int)ax5043ReadReg(0x10));
|
|
printf("Frequency A: 0x%x %x %x %x \n",(int)ax5043ReadReg(0x34),(int)ax5043ReadReg(0x35),(int)ax5043ReadReg(0x36),(int)ax5043ReadReg(0x37));
|
|
|
|
/*
|
|
int x;
|
|
for (x = 0; x < 0x20; x++)
|
|
{
|
|
printf("Register %x contents: %x\n",x,(int)ax5043ReadReg(x));
|
|
}
|
|
|
|
printf("Register Dump complete");
|
|
*/
|
|
return;
|
|
}
|
|
// Adds a Morse space to the buffer
|
|
//
|
|
int add_space(uint8_t *msg) {
|
|
msg[0] = 0x00; // a space is 8 bits
|
|
return 1;
|
|
}
|
|
// Adds a Morse dash to the buffer
|
|
//
|
|
int add_dash(uint8_t *msg, int number) {
|
|
int j;
|
|
int counter = 0;
|
|
|
|
for (j=0; j < number; j++) { // a dot is 4 bits, so a dash is 12 bits
|
|
msg[counter++] = 0xff;
|
|
msg[counter++] = 0x0f;
|
|
}
|
|
return counter;
|
|
}
|
|
// Adds a Morse dot to the buffer
|
|
//
|
|
int add_dot(uint8_t *msg, int number) {
|
|
int counter = 0;
|
|
int j;
|
|
for (j=0; j < number; j++) { // a dot is 4 bits
|
|
msg[counter++] = 0x0f;
|
|
}
|
|
return counter;
|
|
}
|