Changeset - c5bc128e44a6
[Not reviewed]
default
0 3 0
Ethan Zonca - 10 years ago 2016-04-02 20:44:19
ez@ethanzonca.com
WSPR and gps now coexist, don't call jtencode_init because it inits the reed-solomon encoder that wspr doesn't even use
3 files changed with 64 insertions and 36 deletions:
0 comments (0 inline, 0 general)
STM32F031G6_FLASH.ld
Show inline comments
 
@@ -26,25 +26,25 @@
 
**
 
*****************************************************************************
 
*/
 

	
 
/* Entry Point */
 
ENTRY(Reset_Handler)
 

	
 
/* Highest address of the user mode stack */
 
_estack = 0x20001000;    /* end of RAM */
 

	
 
/* Generate a link error if heap and stack don't fit into RAM */
 
_Min_Heap_Size = 0;      /* required amount of heap  */
 
_Min_Stack_Size = 0x300; /* required amount of stack */
 
_Min_Stack_Size = 0x200; /* required amount of stack */
 

	
 
/* Specify the memory areas */
 
MEMORY
 
{
 
FLASH (rx)      : ORIGIN = 0x8000000, LENGTH = 32K
 
RAM (xrw)      : ORIGIN = 0x20000000, LENGTH = 4K
 
}
 

	
 
/* Define output sections */
 
SECTIONS
 
{
 
  /* The startup code goes first into FLASH */
lib/jtencode/jtencode.c
Show inline comments
 
@@ -48,24 +48,37 @@ void jt4_merge_sync_vector(uint8_t *, ui
 
void wspr_merge_sync_vector(uint8_t *, uint8_t *);
 
void convolve(uint8_t *, uint8_t *, uint8_t, uint8_t);
 
void rs_encode(uint8_t *, uint8_t *);
 
void encode_rs_int(void *,data_t *, data_t *);
 
void free_rs_int(void *);
 
void * init_rs_int(int, int, int, int, int, int);
 
void * rs_inst;
 
char callsign[7];
 
char locator[5];
 
uint8_t power;
 

	
 

	
 
static void emz_memcpy( uint8_t *pDest, uint8_t *pSrc, uint32_t len )
 
{
 
    uint32_t i;
 

	
 
    // Manually copy the data
 
    for ( i = 0; i < len; i++ )
 
    {
 
        // Copy data from source to destination
 
        *pDest++ = *pSrc++;
 
    }
 
}
 

	
 

	
 
/* Public Class Members */
 

	
 
void* jtencode_init(void)
 
{
 
  // Initialize the Reed-Solomon encoder
 
  rs_inst = (struct rs *)(intptr_t)init_rs_int(6, 0x43, 3, 1, 51, 0);
 
}
 

	
 
/*
 
 * jt65_encode(char * message, uint8_t * symbols)
 
 *
 
 * Takes an arbitrary message of up to 13 allowable characters and returns
 
@@ -154,58 +167,59 @@ void jt9_encode(char * message, uint8_t 
 
 * jt9_encode(char * message, uint8_t * symbols)
 
 *
 
 * Takes an arbitrary message of up to 13 allowable characters and returns
 
 * a channel symbol table.
 
 *
 
 * message - Plaintext Type 6 message.
 
 * symbols - Array of channel symbols to transmit retunred by the method.
 
 *  Ensure that you pass a uint8_t array of size JT9_SYMBOL_COUNT to the method.
 
 *
 
 */
 
void jt4_encode(char * message, uint8_t * symbols)
 
{
 
    // emz comment memmove
 
  // Ensure that the message text conforms to standards
 
  // --------------------------------------------------
 
  jt_message_prep(message);
 

	
 
  // Bit packing
 
  // -----------
 
  uint8_t c[13];
 
  jt9_bit_packing(message, c);
 

	
 
  // Convolutional Encoding
 
  // ---------------------
 
  uint8_t s[JT4_SYMBOL_COUNT];
 
  convolve(c, s, 13, JT4_BIT_COUNT);
 

	
 
  // Interleaving
 
  // ------------
 
  jt9_interleave(s);
 
  memmove(s + 1, s, JT4_BIT_COUNT);
 
  s[0] = 0; // Append a 0 bit to start of sequence
 

	
 
  // Merge with sync vector
 
  // ----------------------
 
  jt4_merge_sync_vector(s, symbols);
 
//  // --------------------------------------------------
 
//  jt_message_prep(message);
 
//
 
//  // Bit packing
 
//  // -----------
 
//  uint8_t c[13];
 
//  jt9_bit_packing(message, c);
 
//
 
//  // Convolutional Encoding
 
//  // ---------------------
 
//  uint8_t s[JT4_SYMBOL_COUNT];
 
//  convolve(c, s, 13, JT4_BIT_COUNT);
 
//
 
//  // Interleaving
 
//  // ------------
 
//  jt9_interleave(s);
 
//  memmove(s + 1, s, JT4_BIT_COUNT);
 
//  s[0] = 0; // Append a 0 bit to start of sequence
 
//
 
//  // Merge with sync vector
 
//  // ----------------------
 
//  jt4_merge_sync_vector(s, symbols);
 
}
 

	
 
/*
 
 * wspr_encode(char * call, char * loc, uint8_t dbm, uint8_t * symbols)
 
 *
 
 * Takes an arbitrary message of up to 13 allowable characters and returns
 
 *
 
 * call - Callsign (6 characters maximum).
 
 * loc - Maidenhead grid locator (4 charcters maximum).
 
 * loc - Maidenhead grid locator (4 characters maximum).
 
 * dbm - Output power in dBm.
 
 * symbols - Array of channel symbols to transmit retunred by the method.
 
 * symbols - Array of channel symbols to transmit returned by the method.
 
 *  Ensure that you pass a uint8_t array of size WSPR_SYMBOL_COUNT to the method.
 
 *
 
 */
 
void wspr_encode(char * call, char * loc, uint8_t dbm, uint8_t * symbols)
 
{
 
    // Ensure that the message text conforms to standards
 
  // --------------------------------------------------
 
  wspr_message_prep(call, loc, dbm);
 

	
 
  // Bit packing
 
  // -----------
 
  uint8_t c[11];
 
@@ -323,60 +337,70 @@ void jt_message_prep(char * message)
 
}
 

	
 
void wspr_message_prep(char * call, char * loc, uint8_t dbm)
 
{
 
  // Callsign validation and padding
 
  // -------------------------------
 

	
 
	// If only the 2nd character is a digit, then pad with a space.
 
	// If this happens, then the callsign will be truncated if it is
 
	// longer than 5 characters.
 
	if((call[1] >= '0' && call[1] <= '9') && (call[2] < '0' || call[2] > '9'))
 
	{
 
		memmove(call + 1, call, 5);
 
		//memmove(call + 1, call, 5);
 
		//call[0] = ' ';
 

	
 
	    call[6] = '\0';
 
		call[5] = call[4];
 
		call[4] = call[3];
 
		call[3] = call[2];
 
		call[2] = call[1];
 
		call[1] = call[0];
 
		call[0] = ' ';
 

	
 

	
 
	}
 

	
 
	// Now the 3rd charcter in the callsign must be a digit
 
	if(call[2] < '0' || call[2] > '9')
 
	{
 
    // TODO: need a better way to handle this
 
		call[2] = '0';
 
	}
 

	
 
	// Ensure that the only allowed characters are digits and
 
	// uppercase letters
 
	uint8_t i;
 
	for(i = 0; i < 6; i++)
 
	{
 
		call[i] = toupper(call[i]);
 
		if(!(isdigit(call[i]) || isupper(call[i])))
 
		{
 
			call[i] = ' ';
 
		}
 
	}
 

	
 
  memcpy(callsign, call, 6);
 
	emz_memcpy(callsign, call, 6);
 

	
 
	// Grid locator validation
 
	for(i = 0; i < 4; i++)
 
	{
 
		loc[i] = toupper(loc[i]);
 
		if(!(isdigit(loc[i]) || (loc[i] >= 'A' && loc[i] <= 'R')))
 
		{
 
			loc = "AA00";
 
		}
 
	}
 

	
 
  memcpy(locator, loc, 4);
 
	emz_memcpy(locator, loc, 4);
 

	
 
	// Power level validation
 
	// Only certain increments are allowed
 
	if(dbm > 60)
 
	{
 
		dbm = 60;
 
	}
 
  const uint8_t valid_dbm[19] =
 
    {0, 3, 7, 10, 13, 17, 20, 23, 27, 30, 33, 37, 40,
 
     43, 47, 50, 53, 57, 60};
 
  for(i = 0; i < 19; i++)
 
  {
 
@@ -552,25 +576,25 @@ void jt65_interleave(uint8_t * s)
 
    }
 
  }
 

	
 
  // Interleave and translate back to 1D destination array
 
  for(i = 0; i < 7; i++)
 
  {
 
    for(j = 0; j < 9; j++)
 
    {
 
      d[(i * 9) + j] = d1[j][i];
 
    }
 
  }
 

	
 
  memcpy(s, d, JT65_ENCODE_COUNT);
 
  emz_memcpy(s, d, JT65_ENCODE_COUNT);
 
}
 

	
 
void jt9_interleave(uint8_t * s)
 
{
 
  uint8_t i, j, k, n;
 
  uint8_t d[JT9_BIT_COUNT];
 
  uint8_t j0[JT9_BIT_COUNT];
 

	
 
  k = 0;
 

	
 
  // Build the interleave table
 
  for(i = 0; i < 255; i++)
 
@@ -591,25 +615,25 @@ void jt9_interleave(uint8_t * s)
 
    if(k >= 206)
 
    {
 
      break;
 
    }
 
  }
 

	
 
  // Now do the interleave
 
  for(i = 0; i < 206; i++)
 
  {
 
    d[j0[i]] = s[i];
 
  }
 

	
 
  memcpy(s, d, JT9_BIT_COUNT);
 
  emz_memcpy(s, d, JT9_BIT_COUNT);
 
}
 

	
 
void wspr_interleave(uint8_t * s)
 
{
 
  uint8_t d[WSPR_BIT_COUNT];
 
	uint8_t rev, index_temp, i, j, k;
 

	
 
	i = 0;
 

	
 
	for(j = 0; j < 255; j++)
 
	{
 
		// Bit reverse the index
 
@@ -628,32 +652,37 @@ void wspr_interleave(uint8_t * s)
 
		if(rev < WSPR_BIT_COUNT)
 
		{
 
			d[rev] = s[i];
 
			i++;
 
		}
 

	
 
		if(i >= WSPR_BIT_COUNT)
 
		{
 
			break;
 
		}
 
	}
 

	
 
  memcpy(s, d, WSPR_BIT_COUNT);
 
	emz_memcpy(s, d, WSPR_BIT_COUNT);
 
}
 

	
 
void jt9_packbits(uint8_t * d, uint8_t * a)
 
{
 
  uint8_t i, k;
 
  k = 0;
 
  memset(a, 0, JT9_ENCODE_COUNT);
 

	
 
  //memset(a, 0, JT9_ENCODE_COUNT);
 
  for(uint8_t iter = 0; iter<JT9_ENCODE_COUNT; iter++)
 
  {
 
      a[iter] = 0;
 
  }
 

	
 
  for(i = 0; i < JT9_ENCODE_COUNT; i++)
 
  {
 
    a[i] = (d[k] & 1) << 2;
 
    k++;
 

	
 
    a[i] |= ((d[k] & 1) << 1);
 
    k++;
 

	
 
    a[i] |= (d[k] & 1);
 
    k++;
 
  }
src/main.c
Show inline comments
 
@@ -88,26 +88,25 @@ void encode_wspr(void)
 
TIM_HandleTypeDef htim1;
 
 
int main(void)
 
{
 
    HAL_Init();
 
 
    sysclk_init();
 
    gpio_init();
 
    adc_init();
 
    i2c_init();
 
    gps_init();
 
 
 
//    jtencode_init();
 
    //jtencode_init();
 
 
    HAL_Delay(300);
 
 
    // Turn GPS on
 
    HAL_GPIO_WritePin(GPS_NOTEN, 0);
 
 
    // Disable ICs
 
    HAL_GPIO_WritePin(OSC_NOTEN, 1);
 
    HAL_GPIO_WritePin(TCXO_EN, 0);
 
 
    HAL_GPIO_TogglePin(LED_BLUE);
 
 
@@ -118,41 +117,41 @@ int main(void)
 
    htim1.Init.CounterMode = TIM_COUNTERMODE_UP;
 
    htim1.Init.Period = ctc; // Count up to this value (how many 64uS ticks per symbol)
 
    htim1.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
 
    htim1.Init.RepetitionCounter = 0;
 
    HAL_TIM_Base_Init(&htim1);
 
    HAL_TIM_Base_Start_IT(&htim1);
 
    HAL_NVIC_SetPriority(TIM1_BRK_UP_TRG_COM_IRQn, 0, 0);
 
    HAL_NVIC_EnableIRQ(TIM1_BRK_UP_TRG_COM_IRQn);
 
 
 
    uint32_t led_timer = HAL_GetTick();
 
    uint32_t last_gps  = HAL_GetTick();
 
    uint32_t last_wspr  = 0xfffff; // start immediately.
 
    uint32_t last_wspr  = HAL_GetTick(); //0xfffff; // start immediately.
 
 
    HAL_GPIO_TogglePin(LED_BLUE);
 
    HAL_Delay(100);
 
    HAL_GPIO_TogglePin(LED_BLUE);
 
    HAL_Delay(100);
 
    HAL_GPIO_TogglePin(LED_BLUE);
 
    HAL_Delay(100);
 
    HAL_GPIO_TogglePin(LED_BLUE);
 
    HAL_Delay(100);
 
 
 
    while (1)
 
    {
 
        if(HAL_GetTick() - last_wspr > 120000)
 
        {
 
            //encode_wspr();
 
            encode_wspr();
 
            last_wspr = HAL_GetTick();
 
        }
 
 
        if(HAL_GetTick() - led_timer > 100)
 
        {
 
            HAL_GPIO_TogglePin(LED_BLUE);
 
            led_timer = HAL_GetTick();
 
        }
 
        if(HAL_GetTick() - last_gps > 10)
 
        {
 
            parse_gps_transmission();
 
            last_gps = HAL_GetTick();
0 comments (0 inline, 0 general)