mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-24 17:40:24 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			164 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			164 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2010 DSD Author
 | |
|  * GPG Key ID: 0x3F1D7FD0 (74EF 430D F7F2 0A48 FCE6  F630 FAA2 635D 3F1D 7FD0)
 | |
|  *
 | |
|  * Permission to use, copy, modify, and/or distribute this software for any
 | |
|  * purpose with or without fee is hereby granted, provided that the above
 | |
|  * copyright notice and this permission notice appear in all copies.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
 | |
|  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
 | |
|  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
 | |
|  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
 | |
|  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
 | |
|  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
 | |
|  * PERFORMANCE OF THIS SOFTWARE.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * Note: D-STAR support is fairly complete at this point.
 | |
|  * The ambe3600x2450 decoder is similar butnot compatible with D-STAR voice frames. 
 | |
|  * The dstar interleave pattern is different as well.
 | |
|  * GMSK modulation optimizations will also required to get a usable bit error
 | |
|  */
 | |
| 
 | |
| #include "dsd.h"
 | |
| #include "dstar_const.h"
 | |
| #include "dstar_header.h"
 | |
| 
 | |
| 
 | |
| void processDSTAR(dsd_opts * opts, dsd_state * state) {
 | |
| 	// extracts AMBE frames from D-STAR voice frame
 | |
| 	int i, j, dibit;
 | |
| 	char ambe_fr[4][24];
 | |
| 	unsigned char data[9];
 | |
| 	unsigned int bits[4];
 | |
| 	int framecount;
 | |
| 	int sync_missed = 0;
 | |
| 	unsigned char slowdata[4];
 | |
| 	unsigned int bitbuffer = 0;
 | |
| 	const int *w, *x;
 | |
| 
 | |
| 	if (opts->errorbars == 1) {
 | |
| 		printf("e:");
 | |
| 	}
 | |
| 
 | |
| #ifdef DSTAR_DUMP
 | |
| 	printf ("\n");
 | |
| #endif
 | |
| 
 | |
| 	if (state->synctype == 18) {
 | |
| 		framecount = 0;
 | |
| 		state->synctype = 6;
 | |
| 	} else if (state->synctype == 19) {
 | |
| 		framecount = 0;
 | |
| 		state->synctype = 7;
 | |
| 	} else {
 | |
| 		framecount = 1; //just saw a sync frame; there should be 20 not 21 till the next
 | |
| 	}
 | |
| 
 | |
| 	while (sync_missed < 3) {
 | |
| 
 | |
| 		memset(ambe_fr, 0, 96);
 | |
| 		// voice frame
 | |
| 	    w = dW;
 | |
| 	    x = dX;
 | |
| 
 | |
| 		for (i = 0; i < 72; i++) {
 | |
| 
 | |
| 			dibit = getDibit(opts, state);
 | |
| 
 | |
| 			bitbuffer <<= 1;
 | |
| 			if (dibit == 1) {
 | |
| 				bitbuffer |= 0x01;
 | |
| 			}
 | |
| 			if ((bitbuffer & 0x00FFFFFF) == 0x00AAB468) {
 | |
| 				// we're slipping bits
 | |
| 				printf("sync in voice after i=%d, restarting\n", i);
 | |
| 				//ugh just start over
 | |
| 				i = 0;
 | |
| 			    w = dW;
 | |
| 			    x = dX;
 | |
| 				framecount = 1;
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			ambe_fr[*w][*x] = (1 & dibit);
 | |
| 			w++;
 | |
| 			x++;
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		processMbeFrame(opts, state, NULL, ambe_fr, NULL);
 | |
| 
 | |
| 		//  data frame - 24 bits
 | |
| 		for (i = 73; i < 97; i++) {
 | |
| 			dibit = getDibit(opts, state);
 | |
| 			bitbuffer <<= 1;
 | |
| 			if (dibit == 1) {
 | |
| 				bitbuffer |= 0x01;
 | |
| 			}
 | |
| 			if ((bitbuffer & 0x00FFFFFF) == 0x00AAB468) {
 | |
| 				// looking if we're slipping bits
 | |
| 				if (i != 96) {
 | |
| 					printf("sync after i=%d\n", i);
 | |
| 					i = 96;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		slowdata[0] = (bitbuffer >> 16) & 0x000000FF;
 | |
| 		slowdata[1] = (bitbuffer >> 8) & 0x000000FF;
 | |
| 		slowdata[2] = (bitbuffer) & 0x000000FF;
 | |
| 		slowdata[3] = 0;
 | |
| 
 | |
| 		if ((bitbuffer & 0x00FFFFFF) == 0x00AAB468) {
 | |
| 			//We got sync!
 | |
| 			//printf("Sync on framecount = %d\n", framecount);
 | |
| 			sync_missed = 0;
 | |
| 		} else if ((bitbuffer & 0x00FFFFFF) == 0xAAAAAA) {
 | |
| 			//End of transmission
 | |
| 			printf("End of transmission\n");
 | |
| 			goto end;
 | |
| 		} else if (framecount % 21 == 0) {
 | |
| 			printf("Missed sync on framecount = %d, value = %x/%x/%x\n",
 | |
| 					framecount, slowdata[0], slowdata[1], slowdata[2]);
 | |
| 			sync_missed++;
 | |
| 		} else if (framecount != 0 && (bitbuffer & 0x00FFFFFF) != 0x000000) {
 | |
| 			slowdata[0] ^= 0x70;
 | |
| 			slowdata[1] ^= 0x4f;
 | |
| 			slowdata[2] ^= 0x93;
 | |
| 			//printf("unscrambled- %s",slowdata);
 | |
| 
 | |
| 		} else if (framecount == 0) {
 | |
| 			//printf("never scrambled-%s\n",slowdata);
 | |
| 		}
 | |
| 
 | |
| 		framecount++;
 | |
| 	}
 | |
| 
 | |
| 	end: if (opts->errorbars == 1) {
 | |
| 		printf("\n");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void processDSTAR_HD(dsd_opts * opts, dsd_state * state) {
 | |
| 
 | |
| 	int i, j;
 | |
| 	int radioheaderbuffer[660];
 | |
| 
 | |
| 	for (j = 0; j < 660; j++) {
 | |
| 					radioheaderbuffer[j] = getDibit(opts, state);
 | |
| 			}
 | |
| 
 | |
| 	// Note: These routines contain GPLed code. Remove if you object to that.
 | |
| 	// Due to this, they are in a separate source file.
 | |
| 	dstar_header_decode(radioheaderbuffer);
 | |
| 
 | |
| 	//We officially have sync now, so just pass on to the above routine:
 | |
| 
 | |
| 	processDSTAR(opts, state);
 | |
| 
 | |
| }
 | |
| 
 |