500 lines
12 KiB
C
500 lines
12 KiB
C
/* main function
|
|
*
|
|
* (C) 2017 by Andreas Eversberg <jolly@eversberg.eu>
|
|
* All Rights Reserved
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
enum paging_signal;
|
|
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <getopt.h>
|
|
#include <signal.h>
|
|
#include <math.h>
|
|
#include "../libsample/sample.h"
|
|
#include "../libfilter/iir_filter.h"
|
|
#include "../libfm/fm.h"
|
|
#include "../libwave/wave.h"
|
|
#include "../libimage/img.h"
|
|
#include "../libdebug/debug.h"
|
|
#ifdef HAVE_SDR
|
|
#include "../libsdr/sdr_config.h"
|
|
#include "../libsdr/sdr.h"
|
|
#endif
|
|
#include "bas.h"
|
|
#include "tv_modulate.h"
|
|
#include "channels.h"
|
|
|
|
void *sender_head = NULL;
|
|
int use_sdr = 0;
|
|
int num_kanal = 1; /* only one channel used for debugging */
|
|
|
|
void clear_console_text() {}
|
|
void print_console_text() {}
|
|
void display_status_limit_scroll() {}
|
|
void *get_sender_by_empfangsfrequenz() { return NULL; }
|
|
|
|
static double __attribute__((__unused__)) modulation = 0.7; /* level of modulation for I/Q amplitudes */
|
|
static double frequency = 0.0;
|
|
static int fbas = 1;
|
|
static int tone = 1;
|
|
static double circle_radius = 6.7;
|
|
static int color_bar = 0;
|
|
static int grid_only = 0;
|
|
static const char *station_id = "Jolly Roger";
|
|
static int __attribute__((__unused__)) latency = 30;
|
|
static double samplerate = 10e6;
|
|
static const char *wave_file = NULL;
|
|
|
|
/* global variable to quit main loop */
|
|
int quit = 0;
|
|
|
|
void sighandler(int sigset)
|
|
{
|
|
if (sigset == SIGHUP)
|
|
return;
|
|
if (sigset == SIGPIPE)
|
|
return;
|
|
|
|
// clear_console_text();
|
|
printf("Signal received: %d\n", sigset);
|
|
|
|
quit = 1;
|
|
}
|
|
|
|
void print_help(const char *arg0)
|
|
|
|
{
|
|
printf("Usage: %s -f <frequency> | -c <channel> <command>\n", arg0);
|
|
/* - - */
|
|
printf("\ncommand:\n");
|
|
printf(" tx-fubk Transmit FUBK test image (German PAL image)\n");
|
|
printf(" tx-vcr Transmit VCR calibration pattern\n");
|
|
printf(" tx-img <image> Transmit given image file\n");
|
|
printf(" Use 4:3 image with 574 lines for best result.\n");
|
|
printf("\ngeneral options:\n");
|
|
printf(" -f --frequency <frequency>\n");
|
|
printf(" Give frequency in Hertz.\n");
|
|
printf(" -c --channel <channel>\n");
|
|
printf(" Or give channel number.\n");
|
|
printf(" Use 'list' to get a channel list.\n");
|
|
printf(" -r --samplerate <frequency>\n");
|
|
printf(" Give sample rate in Hertz.\n");
|
|
printf(" -w --wave-file <filename>\n");
|
|
printf(" Output to wave file instead of SDR\n");
|
|
printf("\nsignal options:\n");
|
|
printf(" -F --fbas 1 | 0\n");
|
|
printf(" Turn color on or off. (default = %d)\n", fbas);
|
|
printf(" -T --tone 1 | 0\n");
|
|
printf(" Turn tone on or off. (default = %d)\n", tone);
|
|
printf("\nFUBK options:\n");
|
|
printf(" -R --circle-radius <radius> | 0\n");
|
|
printf(" Use circle radius or 0 for off. (default = %.1f)\n", circle_radius);
|
|
printf(" -C --color-bar 1 | 0\n");
|
|
printf(" 1 == Color bar on, 0 = Show complete middle field. (default = %d)\n", color_bar);
|
|
printf(" For clean scope signal, also turn off circle by setting radius to 0.\n");
|
|
printf(" -G --grid-only 1 | 0\n");
|
|
printf(" 1 == Show only the grid of the test image. (default = %d)\n", grid_only);
|
|
printf(" -I --sation-id \"<text>\"\n");
|
|
printf(" Give exactly 12 characters to display as Station ID.\n");
|
|
printf(" (default = \"%s\")\n", station_id);
|
|
#ifdef HAVE_SDR
|
|
sdr_config_print_help();
|
|
#endif
|
|
}
|
|
|
|
static struct option long_options_common[] = {
|
|
{"help", 0, 0, 'h'},
|
|
{"frequency", 1, 0, 'f'},
|
|
{"channel", 1, 0, 'c'},
|
|
{"samplerate", 1, 0, 'r'},
|
|
{"wave-file", 1, 0, 'w'},
|
|
{"fbas", 1, 0, 'F'},
|
|
{"tone", 1, 0, 'T'},
|
|
{"circle-radius", 1, 0, 'R'},
|
|
{"color-bar", 1, 0, 'C'},
|
|
{"grid-only", 1, 0, 'G'},
|
|
{"station-id", 1, 0, 'I'},
|
|
{0, 0, 0, 0}
|
|
};
|
|
|
|
static const char *optstring_common = "hf:c:r:w:F:T:R:C:G:I:";
|
|
|
|
struct option *long_options;
|
|
char *optstring;
|
|
|
|
static void check_duplicate_option(int num, struct option *option)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < num; i++) {
|
|
if (long_options[i].val == option->val) {
|
|
fprintf(stderr, "Duplicate option %d. Please fix!\n", option->val);
|
|
abort();
|
|
}
|
|
}
|
|
}
|
|
|
|
void set_options_common(void)
|
|
{
|
|
int i = 0, j;
|
|
|
|
long_options = calloc(sizeof(*long_options), 256);
|
|
for (j = 0; long_options_common[i].name; i++, j++) {
|
|
check_duplicate_option(i, &long_options_common[j]);
|
|
memcpy(&long_options[i], &long_options_common[j], sizeof(*long_options));
|
|
}
|
|
#ifdef HAVE_SDR
|
|
for (j = 0; sdr_config_long_options[j].name; i++, j++) {
|
|
check_duplicate_option(i, &sdr_config_long_options[j]);
|
|
memcpy(&long_options[i], &sdr_config_long_options[j], sizeof(*long_options));
|
|
}
|
|
#endif
|
|
|
|
optstring = calloc(256, 2);
|
|
strcpy(optstring, optstring_common);
|
|
#ifdef HAVE_SDR
|
|
strcat(optstring, sdr_config_optstring);
|
|
#endif
|
|
}
|
|
|
|
static int handle_options(int argc, char **argv)
|
|
{
|
|
int skip_args = 0;
|
|
#ifdef HAVE_SDR
|
|
int rc;
|
|
#endif
|
|
|
|
set_options_common();
|
|
|
|
while (1) {
|
|
int option_index = 0, c;
|
|
|
|
c = getopt_long(argc, argv, optstring, long_options, &option_index);
|
|
|
|
if (c == -1)
|
|
break;
|
|
|
|
switch (c) {
|
|
case 'h':
|
|
print_help(argv[0]);
|
|
exit(0);
|
|
case 'f':
|
|
frequency = atof(optarg);
|
|
skip_args += 2;
|
|
break;
|
|
case 'c':
|
|
if (!strcmp(optarg, "list")) {
|
|
list_tv_channels();
|
|
exit(0);
|
|
}
|
|
frequency = get_tv_video_frequency(atoi(optarg));
|
|
if (frequency == 0.0) {
|
|
fprintf(stderr, "Given channel number unknown, use \"-c list\" to get a list.\n");
|
|
exit(0);
|
|
}
|
|
skip_args += 2;
|
|
break;
|
|
case 'r':
|
|
samplerate = atof(optarg);
|
|
skip_args += 2;
|
|
break;
|
|
case 'w':
|
|
wave_file = strdup(optarg);
|
|
skip_args += 2;
|
|
break;
|
|
case 'F':
|
|
fbas = atoi(optarg);
|
|
skip_args += 2;
|
|
break;
|
|
case 'T':
|
|
tone = atoi(optarg);
|
|
skip_args += 2;
|
|
break;
|
|
case 'R':
|
|
circle_radius = atof(optarg);
|
|
skip_args += 2;
|
|
break;
|
|
case 'C':
|
|
color_bar = atoi(optarg);
|
|
skip_args += 2;
|
|
break;
|
|
case 'G':
|
|
grid_only = atoi(optarg);
|
|
skip_args += 2;
|
|
break;
|
|
case 'I':
|
|
station_id = strdup(optarg);
|
|
if (strlen(station_id) != 12) {
|
|
fprintf(stderr, "Given station ID must be exactly 12 charaters long. (Use spaces to fill it.)\n");
|
|
exit(0);
|
|
}
|
|
skip_args += 2;
|
|
break;
|
|
default:
|
|
#ifdef HAVE_SDR
|
|
rc = sdr_config_opt_switch(c, &skip_args);
|
|
if (rc < 0)
|
|
exit(0);
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
|
|
return skip_args;
|
|
}
|
|
|
|
static void tx_bas(sample_t *sample_bas, __attribute__((__unused__)) sample_t *sample_tone, __attribute__((__unused__)) uint8_t *power_tone, int samples)
|
|
{
|
|
/* catch signals */
|
|
signal(SIGINT, sighandler);
|
|
signal(SIGHUP, sighandler);
|
|
signal(SIGTERM, sighandler);
|
|
signal(SIGPIPE, sighandler);
|
|
|
|
if (wave_file) {
|
|
wave_rec_t rec;
|
|
int rc;
|
|
sample_t *buffers[1];
|
|
|
|
rc = wave_create_record(&rec, wave_file, samplerate, 1, 1.0);
|
|
if (rc < 0) {
|
|
// FIXME cleanup
|
|
exit(0);
|
|
}
|
|
|
|
buffers[0] = sample_bas;
|
|
wave_write(&rec, buffers, samples);
|
|
|
|
wave_destroy_record(&rec);
|
|
} else {
|
|
#ifdef HAVE_SDR
|
|
float *buff = NULL;
|
|
void *sdr = NULL;
|
|
int latspl = samplerate * latency / 1000;
|
|
float *sendbuff = NULL;
|
|
|
|
if ((sdr_config->uhd == 0 && sdr_config->soapy == 0)) {
|
|
fprintf(stderr, "You must choose SDR API you want: --sdr-uhd or --sdr-soapy or -w <file> to generate wave file.\n");
|
|
goto error;
|
|
}
|
|
|
|
sendbuff = calloc(latspl * 2, sizeof(*sendbuff));
|
|
if (!sendbuff) {
|
|
fprintf(stderr, "No mem!\n");
|
|
goto error;
|
|
}
|
|
|
|
/* modulate */
|
|
buff = calloc(samples + 10.0, sizeof(sample_t) * 2);
|
|
if (!buff) {
|
|
fprintf(stderr, "No mem!\n");
|
|
goto error;
|
|
}
|
|
tv_modulate(buff, samples, sample_bas, modulation);
|
|
|
|
if (sample_tone) {
|
|
/* bandwidth is 2*(deviation + 2*f(sig)) = 2 * (50 + 2*15) = 160khz */
|
|
fm_mod_t mod;
|
|
fm_mod_init(&mod, samplerate, 5500000.0, modulation * 0.1);
|
|
mod.state = MOD_STATE_ON; /* do not ramp up */
|
|
fm_modulate_complex(&mod, sample_tone, power_tone, samples, buff);
|
|
}
|
|
|
|
double tx_frequencies[1], rx_frequencies[1];
|
|
tx_frequencies[0] = frequency;
|
|
rx_frequencies[0] = frequency;
|
|
sdr = sdr_open(NULL, tx_frequencies, rx_frequencies, 1, 0.0, samplerate, latspl, 0.0, 0.0);
|
|
if (!sdr)
|
|
goto error;
|
|
sdr_start(sdr);
|
|
|
|
int pos = 0, max = samples * 2;
|
|
int s, ss, tosend;
|
|
while (!quit) {
|
|
usleep(1000);
|
|
sdr_read(sdr, (void *)sendbuff, latspl, 0, NULL);
|
|
tosend = sdr_get_tosend(sdr, latspl);
|
|
if (tosend > latspl / 10)
|
|
tosend = latspl / 10;
|
|
if (tosend == 0) {
|
|
continue;
|
|
}
|
|
for (s = 0, ss = 0; s < tosend; s++) {
|
|
sendbuff[ss++] = buff[pos++];
|
|
sendbuff[ss++] = buff[pos++];
|
|
if (pos == max) {
|
|
pos = 0;
|
|
}
|
|
}
|
|
sdr_write(sdr, (void *)sendbuff, NULL, tosend, NULL, NULL, 0);
|
|
}
|
|
|
|
error:
|
|
free(sendbuff);
|
|
free(buff);
|
|
if (sdr)
|
|
sdr_close(sdr);
|
|
#endif
|
|
}
|
|
|
|
/* reset signals */
|
|
signal(SIGINT, SIG_DFL);
|
|
signal(SIGHUP, SIG_DFL);
|
|
signal(SIGTERM, SIG_DFL);
|
|
signal(SIGPIPE, SIG_DFL);
|
|
}
|
|
|
|
static int tx_test_picture(enum bas_type type)
|
|
{
|
|
bas_t bas;
|
|
sample_t *test_bas = NULL;
|
|
sample_t *test_tone = NULL;
|
|
uint8_t *test_power = NULL;
|
|
int i;
|
|
int ret = -1;
|
|
int count;
|
|
|
|
/* test image, add some samples in case of overflow due to rounding errors */
|
|
test_bas = calloc(samplerate / 25.0 * 4.0 + 10.0, sizeof(sample_t));
|
|
if (!test_bas) {
|
|
fprintf(stderr, "No mem!\n");
|
|
goto error;
|
|
}
|
|
bas_init(&bas, samplerate, type, fbas, circle_radius, color_bar, grid_only, station_id, NULL, 0, 0);
|
|
count = bas_generate(&bas, test_bas);
|
|
count += bas_generate(&bas, test_bas + count);
|
|
count += bas_generate(&bas, test_bas + count);
|
|
count += bas_generate(&bas, test_bas + count);
|
|
|
|
if (tone) {
|
|
/* for more about audio modulation on tv, see: http://elektroniktutor.de/geraetetechnik/fston.html */
|
|
test_tone = calloc(count, sizeof(sample_t));
|
|
if (!test_tone) {
|
|
fprintf(stderr, "No mem!\n");
|
|
goto error;
|
|
}
|
|
test_power = calloc(count, sizeof(uint8_t));
|
|
if (!test_power) {
|
|
fprintf(stderr, "No mem!\n");
|
|
goto error;
|
|
}
|
|
/* emphasis 50us, but 1000Hz does not change level */
|
|
for (i = 0; i < count; i++)
|
|
test_tone[i] = sin((double)i * 2.0 * M_PI * 1000.0 / samplerate) * 50000;
|
|
memset(test_power, 1, count);
|
|
}
|
|
|
|
tx_bas(test_bas, test_tone, test_power, count);
|
|
|
|
ret = 0;
|
|
error:
|
|
free(test_bas);
|
|
free(test_tone);
|
|
free(test_power);
|
|
return ret;
|
|
}
|
|
|
|
static int tx_img(const char *filename)
|
|
{
|
|
unsigned short *img = NULL;
|
|
int width, height;
|
|
bas_t bas;
|
|
sample_t *img_bas = NULL;
|
|
int ret = -1;
|
|
int count;
|
|
|
|
img = load_img(&width, &height, filename, 0);
|
|
if (!img) {
|
|
fprintf(stderr, "Failed to load grey image '%s'\n", filename);
|
|
return -1;
|
|
}
|
|
|
|
/* test image, add some samples in case of overflow due to rounding errors */
|
|
img_bas = calloc(samplerate / 25.0 * 4.0 + 10.0, sizeof(sample_t));
|
|
if (!img_bas) {
|
|
fprintf(stderr, "No mem!\n");
|
|
goto error;
|
|
}
|
|
bas_init(&bas, samplerate, BAS_IMAGE, fbas, circle_radius, color_bar, grid_only, NULL, img, width, height);
|
|
count = bas_generate(&bas, img_bas);
|
|
count += bas_generate(&bas, img_bas + count);
|
|
count += bas_generate(&bas, img_bas + count);
|
|
count += bas_generate(&bas, img_bas + count);
|
|
|
|
tx_bas(img_bas, NULL, NULL, count);
|
|
|
|
ret = 0;
|
|
error:
|
|
free(img_bas);
|
|
free(img);
|
|
return ret;
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
int skip_args;
|
|
int __attribute__((__unused__)) rc;
|
|
const char *arg0 = argv[0];
|
|
|
|
debuglevel = 0;
|
|
|
|
#ifdef HAVE_SDR
|
|
sdr_config_init();
|
|
#endif
|
|
|
|
skip_args = handle_options(argc, argv);
|
|
argc -= skip_args + 1;
|
|
argv += skip_args + 1;
|
|
|
|
if (frequency == 0.0 && !wave_file) {
|
|
print_help(arg0);
|
|
exit(0);
|
|
}
|
|
|
|
if (!wave_file) {
|
|
#ifdef HAVE_SDR
|
|
rc = sdr_configure(samplerate);
|
|
if (rc < 0)
|
|
return rc;
|
|
#endif
|
|
}
|
|
|
|
if (argc < 1) {
|
|
fprintf(stderr, "Expecting command, see help!\n");
|
|
exit(0);
|
|
} else if (!strcmp(argv[0], "tx-fubk")) {
|
|
tx_test_picture(BAS_FUBK);
|
|
} else if (!strcmp(argv[0], "tx-vcr")) {
|
|
tx_test_picture(BAS_VCR);
|
|
} else if (!strcmp(argv[0], "tx-img")) {
|
|
if (argc < 2) {
|
|
fprintf(stderr, "Expecting image file, see help!\n");
|
|
exit(0);
|
|
}
|
|
tx_img(argv[1]);
|
|
} else {
|
|
fprintf(stderr, "Unknown command '%s', see help!\n", argv[0]);
|
|
exit(0);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|