Files

copied
Last update 2 years 1 month by Gianpaolo Macario
Fileshosthackrf-toolssrc
..
CMakeLists.txt
hackrf_clock.c
hackrf_cpldjtag.c
hackrf_debug.c
hackrf_info.c
hackrf_operacake.c
hackrf_spiflash.c
hackrf_sweep.c
hackrf_transfer.c
hackrf_operacake.c
/* * Copyright 2016-2022 Great Scott Gadgets <info@greatscottgadgets.com> * Copyright 2016 Dominic Spill <dominicgs@gmail.com> * * This file is part of HackRF. * * 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 2, 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; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, * Boston, MA 02110-1301, USA. */ #include <hackrf.h> #include <stdio.h> #include <stdlib.h> #include <getopt.h> #include <string.h> #ifndef bool typedef int bool; #define true 1 #define false 0 #endif #define FREQ_MIN_MHZ (0) /* 0 MHz */ #define FREQ_MAX_MHZ (7250) /* 7250 MHz */ #define INVALID_ADDRESS 0xFF #define INVALID_MODE 0xFF #define INVALID_PORT 0xFF #define GPIO_TEST_DISABLED 0xFFFF static void usage() { printf("\nUsage:\n"); printf("\t-h, --help: this help\n"); printf("\t-d, --device <n>: specify a particular device by serial number\n"); printf("\t-o, --address <n>: specify a particular Opera Cake by address [default: 0]\n"); printf("\t-m, --mode <mode>: specify switching mode [options: manual, frequency, time]\n"); printf("\t-a <port>: set port connected to port A0\n"); printf("\t-b <port>: set port connected to port B0\n"); printf("\t-f <port:min:max>: automatically assign <port> for range <min:max> in MHz. This argument can be repeated to specify a list of ports.\n"); printf("\t-t <port:dwell>: in time mode, dwell on <port> for <dwell> samples. Specify only <port> to use the default dwell time (with -w). This argument can be repeated to specify a list of ports.\n"); printf("\t-w <n>: set default dwell time in samples for time mode\n"); printf("\t-l, --list: list available Opera Cake boards\n"); printf("\t-g, --gpio_test: test GPIO functionality of an Opera Cake\n"); } static struct option long_options[] = { {"device", required_argument, 0, 'd'}, {"address", required_argument, 0, 'o'}, {"mode", required_argument, 0, 'm'}, {"list", no_argument, 0, 'l'}, {"gpio_test", no_argument, 0, 'g'}, {"help", no_argument, 0, 'h'}, {0, 0, 0, 0}, }; int parse_uint16(char* const s, uint16_t* const value) { char* s_end = s; const long long_value = strtol(s, &s_end, 10); if ((s != s_end) && (*s_end == 0)) { *value = (uint16_t) long_value; return HACKRF_SUCCESS; } else { return HACKRF_ERROR_INVALID_PARAM; } } int parse_uint32(char* const s, uint32_t* const value) { char* s_end = s; const long long_value = strtol(s, &s_end, 10); if ((s != s_end) && (*s_end == 0)) { *value = (uint32_t) long_value; return HACKRF_SUCCESS; } else { return HACKRF_ERROR_INVALID_PARAM; } } int parse_port(char* str, uint8_t* port) { uint16_t tmp_port; int result; if (str[0] == 'A' || str[0] == 'B') { // The port was specified as a side and number eg. A1 or B3 result = parse_uint16(str + 1, &tmp_port); if (result != HACKRF_SUCCESS) { return result; } if (tmp_port >= 5 || tmp_port <= 0) { fprintf(stderr, "invalid port: %s\n", str); return HACKRF_ERROR_INVALID_PARAM; } // Value was a valid port between 0-4 if (str[0] == 'A') { // A1=0, A2=1, A3=2, A4=3 tmp_port -= 1; } else { // If B was specfied just add 4-1 ports // B1=4, B2=5, B3=6, B4=7 tmp_port += 3; } } else { result = parse_uint16(str, &tmp_port); if (result != HACKRF_SUCCESS) { return result; } } *port = tmp_port & 0xFF; // printf("Port: %d\n", *port); return HACKRF_SUCCESS; } int parse_range(char* s, hackrf_operacake_freq_range* range) { char port[16]; float min; float max; int result; // Read frequency as a float here to support scientific notation (e.g: 1e6) if (sscanf(s, "%15[^:]:%f:%f", port, &min, &max) == 3) { result = parse_port(port, &(range->port)); if (result != HACKRF_SUCCESS) { return result; } range->freq_min = min; range->freq_max = max; return HACKRF_SUCCESS; } return HACKRF_ERROR_INVALID_PARAM; } int parse_dwell(char* s, hackrf_operacake_dwell_time* dwell_time) { int result; char port[16]; float dwell; // Read dwell as a float here to support scientific notation (e.g: 1e6) if (sscanf(s, "%15[^:]:%f", port, &dwell) == 2) { result = parse_port(port, &dwell_time->port); if (result != HACKRF_SUCCESS) { return result; } if (dwell == 0) { fprintf(stderr, "dwell time cannot be 0\n"); return HACKRF_ERROR_INVALID_PARAM; } dwell_time->dwell = (uint32_t) dwell; return HACKRF_SUCCESS; } else if (sscanf(s, "%15[^:]", port) == 1) { result = parse_port(port, &dwell_time->port); if (result != HACKRF_SUCCESS) { return result; } // This will be replaced with the default dwell time later. dwell_time->dwell = 0; return HACKRF_SUCCESS; } return HACKRF_ERROR_INVALID_PARAM; } int main(int argc, char** argv) { int opt; const char* serial_number = NULL; uint8_t operacake_address = 0; bool set_mode = false; uint8_t mode; uint8_t port_a = INVALID_PORT; uint8_t port_b = INVALID_PORT; bool set_ports = false; bool list = false; bool gpio_test = false; uint8_t operacakes[8]; uint8_t operacake_count = 0; int i = 0; hackrf_device* device = NULL; int option_index = 0; hackrf_operacake_freq_range ranges[HACKRF_OPERACAKE_MAX_FREQ_RANGES]; hackrf_operacake_dwell_time dwell_times[HACKRF_OPERACAKE_MAX_DWELL_TIMES]; uint8_t range_idx = 0; uint8_t dwell_idx = 0; uint32_t default_dwell = 0; int result = hackrf_init(); if (result) { printf("hackrf_init() failed: %s (%d)\n", hackrf_error_name(result), result); return -1; } while ((opt = getopt_long( argc, argv, "d:o:a:m:b:lf:t:w:hg?", long_options, &option_index)) != EOF) { switch (opt) { case 'd': serial_number = optarg; break; case 'o': operacake_address = atoi(optarg); break; case 'm': if (strcmp(optarg, "manual") == 0) { mode = OPERACAKE_MODE_MANUAL; set_mode = true; } else if (strcmp(optarg, "frequency") == 0) { mode = OPERACAKE_MODE_FREQUENCY; set_mode = true; } else if (strcmp(optarg, "time") == 0) { mode = OPERACAKE_MODE_TIME; set_mode = true; } else { fprintf(stderr, "argument error: mode must be one of [manual, frequency, time].\n"); usage(); return EXIT_FAILURE; } break; case 'f': if (HACKRF_OPERACAKE_MAX_FREQ_RANGES == range_idx) { fprintf(stderr, "argument error: specify a maximum of %u frequency ranges.\n", HACKRF_OPERACAKE_MAX_FREQ_RANGES); usage(); return EXIT_FAILURE; } result = parse_range(optarg, &ranges[range_idx]); if (result != HACKRF_SUCCESS) { fprintf(stderr, "failed to parse range\n"); return EXIT_FAILURE; } if (ranges[range_idx].freq_min >= ranges[range_idx].freq_max) { fprintf(stderr, "argument error: freq_max must be greater than freq_min.\n"); usage(); return EXIT_FAILURE; } if (FREQ_MAX_MHZ < ranges[range_idx].freq_max) { fprintf(stderr, "argument error: freq_max may not be higher than %u.\n", FREQ_MAX_MHZ); usage(); return EXIT_FAILURE; } range_idx++; break; case 't': if (HACKRF_OPERACAKE_MAX_DWELL_TIMES == dwell_idx) { fprintf(stderr, "argument error: specify a maximum of %u dwell times.\n", HACKRF_OPERACAKE_MAX_DWELL_TIMES); usage(); return EXIT_FAILURE; } result = parse_dwell(optarg, &dwell_times[dwell_idx]); if (result != HACKRF_SUCCESS) { fprintf(stderr, "failed to parse dwell time\n"); return EXIT_FAILURE; } dwell_idx++; break; case 'w': default_dwell = atof(optarg); break; case 'a': result = parse_port(optarg, &port_a); if (result != HACKRF_SUCCESS) { fprintf(stderr, "failed to parse port\n"); return EXIT_FAILURE; } set_ports = true; break; case 'b': result = parse_port(optarg, &port_b); if (result != HACKRF_SUCCESS) { fprintf(stderr, "failed to parse port\n"); return EXIT_FAILURE; } set_ports = true; break; case 'l': list = true; break; case 'g': gpio_test = true; break; case 'h': case '?': usage(); return EXIT_SUCCESS; default: fprintf(stderr, "unknown argument '-%c %s'\n", opt, optarg); usage(); return EXIT_FAILURE; } } // Any operations that set a parameter on an Opera Cake board. bool set_params = set_mode || set_ports || range_idx || dwell_idx; // Error out unless exactly one option is selected. if (list + set_params + gpio_test != 1) { fprintf(stderr, "Specify either list, mode, or GPIO test option.\n"); usage(); return EXIT_FAILURE; } result = hackrf_open_by_serial(serial_number, &device); if (result != HACKRF_SUCCESS) { fprintf(stderr, "hackrf_open() failed: %s (%d)\n", hackrf_error_name(result), result); return EXIT_FAILURE; } if (set_mode) { result = hackrf_set_operacake_mode(device, operacake_address, mode); if (result != HACKRF_SUCCESS) { fprintf(stderr, "hackrf_set_operacake_mode() failed: %s (%d)\n", hackrf_error_name(result), result); return EXIT_FAILURE; } } if (list) { result = hackrf_get_operacake_boards(device, operacakes); if (result != HACKRF_SUCCESS) { fprintf(stderr, "hackrf_get_operacake_boards() failed: %s (%d)\n", hackrf_error_name(result), result); return EXIT_FAILURE; } printf("Opera Cakes found: "); for (i = 0; i < 8; i++) { if (operacakes[i] != HACKRF_OPERACAKE_ADDRESS_INVALID) { printf("\n\tAddress: %d", operacakes[i]); enum operacake_switching_mode mode; hackrf_get_operacake_mode(device, i, &mode); printf("\tSwitching mode: "); if (mode == OPERACAKE_MODE_MANUAL) { printf("manual\n"); } else if (mode == OPERACAKE_MODE_FREQUENCY) { printf("frequency\n"); } else if (mode == OPERACAKE_MODE_TIME) { printf("time\n"); } else { printf("unknown\n"); } operacake_count++; } } if (!operacake_count) { printf("None"); } printf("\n"); } if (gpio_test) { uint16_t test_result; uint8_t reg, mask = 0x7; result = hackrf_operacake_gpio_test( device, operacake_address, &test_result); if (result != HACKRF_SUCCESS) { fprintf(stderr, "hackrf_operacake_gpio_test() failed: %s (%d)\n", hackrf_error_name(result), result); return EXIT_FAILURE; } if (test_result == GPIO_TEST_DISABLED) { fprintf(stderr, "GPIO mode disabled.\n"); fprintf(stderr, "Remove additional add-on boards and retry.\n"); } else if (test_result) { fprintf(stderr, "GPIO test failed\n"); fprintf(stderr, "Pin\tHigh\tShorts\tLow\n"); reg = test_result & mask; fprintf(stderr, "u2ctrl1\t%d\t%d\t%d\n", (reg >> 2) & 1, (reg >> 1) & 1, reg & 1); test_result >>= 3; reg = test_result & mask; fprintf(stderr, "u2ctrl0\t%d\t%d\t%d\n", (reg >> 2) & 1, (reg >> 1) & 1, reg & 1); test_result >>= 3; reg = test_result & mask; fprintf(stderr, "u3ctrl1\t%d\t%d\t%d\n", (reg >> 2) & 1, (reg >> 1) & 1, reg & 1); test_result >>= 3; reg = test_result & mask; fprintf(stderr, "u3ctrl0\t%d\t%d\t%d\n", (reg >> 2) & 1, (reg >> 1) & 1, reg & 1); test_result >>= 3; reg = test_result & mask; fprintf(stderr, "u1ctrl \t%d\t%d\t%d\n", (reg >> 2) & 1, (reg >> 1) & 1, reg & 1); } else { fprintf(stderr, "GPIO test passed\n"); } } if (set_ports) { // Set other port to "don't care" if not set if (port_a == INVALID_PORT) { if (port_b >= 4) { port_a = 0; } else { port_a = 4; } } if (port_b == INVALID_PORT) { if (port_a >= 4) { port_b = 0; } else { port_b = 4; } } if (((port_a <= 3) && (port_b <= 3)) || ((port_a >= 4) && (port_b >= 4))) { fprintf(stderr, "Port A and B cannot be connected to the same side\n"); return EXIT_FAILURE; } result = hackrf_set_operacake_ports( device, operacake_address, port_a, port_b); if (result) { printf("hackrf_set_operacake_ports() failed: %s (%d)\n", hackrf_error_name(result), result); return EXIT_FAILURE; } } if (range_idx) { result = hackrf_set_operacake_freq_ranges(device, ranges, range_idx); if (result) { printf("hackrf_set_operacake_freq_ranges() failed: %s (%d)\n", hackrf_error_name(result), result); return -1; } } if (dwell_idx) { for (i = 0; i < dwell_idx; i++) { if (dwell_times[i].dwell == 0) { if (default_dwell == 0) { fprintf(stderr, "port '%u' set to use default dwell time, but default dwell time is not set. Use -w argument to set default dwell time.\n", dwell_times[i].port); return EXIT_FAILURE; } dwell_times[i].dwell = default_dwell; } } result = hackrf_set_operacake_dwell_times(device, dwell_times, dwell_idx); if (result) { printf("hackrf_set_operacake_dwell_times() failed: %s (%d)\n", hackrf_error_name(result), result); return -1; } } result = hackrf_close(device); if (result) { printf("hackrf_close() failed: %s (%d)\n", hackrf_error_name(result), result); return -1; } hackrf_exit(); return 0; }
Report a bug