
* Initial commit of hackrf_biast * Cleaned up hackrf_biast * Cleaned up usage info * Include getopt.h for non-GNU systems * Add support for overriding HackRF's default antenna power behavior in firmware. Add support for specifying antenna power behavior in libhackrf. * Moved bias tee config routines into user_config.c, cleaned up operation of hackrf_biast * hackrf_biast now calls usage() and exits when invoked with no arguments * Fixed minor documentation error in usage() * minor syntax cleanup * Add some documentation to the host API call * Add proper declaration magic in hackrf.h to hackrf_set_user_bias_t_opts() to appease Visual Studio * Documentation changes suggested by @martinling * Moved bias t setting above switch statement, removed line that explicit turned bias t off when entering OFF mode * Change hackrf_set_user_bias_t_opts() to use a friendly struct() instead of a bitmask. User friendliness fixes to hackrf_biast options. More clang-format appeasement. * Removed support for integer mode args from hackrf_biast * clang-format error fixes * Tweaked position of comment for clang-format v14 * Reformat files with clang-format v14 instead of 16 * Remove internal numeric modes for bias T settings Co-authored-by: Martin Ling <martin-github@earth.li> * Fix documentation error in hackrf_biast.c --------- Co-authored-by: Martin Ling <martin-github@earth.li>
222 lines
5.7 KiB
C
222 lines
5.7 KiB
C
/*
|
|
* Copyright 2012-2022 Great Scott Gadgets <info@greatscottgadgets.com>
|
|
* Copyright 2012 Jared Boone
|
|
* Copyright 2013 Benjamin Vernoux
|
|
*
|
|
* 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 "usb_api_register.h"
|
|
#include <user_config.h>
|
|
#include <hackrf_core.h>
|
|
#include <usb_queue.h>
|
|
#include <max283x.h>
|
|
#include <rffc5071.h>
|
|
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
|
|
#include <hackrf_core.h>
|
|
|
|
usb_request_status_t usb_vendor_request_write_max283x(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
if (endpoint->setup.index < MAX2837_NUM_REGS) {
|
|
if (endpoint->setup.value < MAX2837_DATA_REGS_MAX_VALUE) {
|
|
max283x_reg_write(
|
|
&max283x,
|
|
endpoint->setup.index,
|
|
endpoint->setup.value);
|
|
usb_transfer_schedule_ack(endpoint->in);
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
}
|
|
return USB_REQUEST_STATUS_STALL;
|
|
} else {
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
}
|
|
|
|
usb_request_status_t usb_vendor_request_read_max283x(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
if (endpoint->setup.index < MAX2837_NUM_REGS) {
|
|
const uint16_t value =
|
|
max283x_reg_read(&max283x, endpoint->setup.index);
|
|
endpoint->buffer[0] = value & 0xff;
|
|
endpoint->buffer[1] = value >> 8;
|
|
usb_transfer_schedule_block(
|
|
endpoint->in,
|
|
&endpoint->buffer,
|
|
2,
|
|
NULL,
|
|
NULL);
|
|
usb_transfer_schedule_ack(endpoint->out);
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
return USB_REQUEST_STATUS_STALL;
|
|
} else {
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
}
|
|
|
|
usb_request_status_t usb_vendor_request_write_si5351c(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
if (endpoint->setup.index < 256) {
|
|
if (endpoint->setup.value < 256) {
|
|
si5351c_write_single(
|
|
&clock_gen,
|
|
endpoint->setup.index,
|
|
endpoint->setup.value);
|
|
usb_transfer_schedule_ack(endpoint->in);
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
}
|
|
return USB_REQUEST_STATUS_STALL;
|
|
} else {
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
}
|
|
|
|
usb_request_status_t usb_vendor_request_read_si5351c(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
if (endpoint->setup.index < 256) {
|
|
const uint8_t value =
|
|
si5351c_read_single(&clock_gen, endpoint->setup.index);
|
|
endpoint->buffer[0] = value;
|
|
usb_transfer_schedule_block(
|
|
endpoint->in,
|
|
&endpoint->buffer,
|
|
1,
|
|
NULL,
|
|
NULL);
|
|
usb_transfer_schedule_ack(endpoint->out);
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
return USB_REQUEST_STATUS_STALL;
|
|
} else {
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
}
|
|
|
|
#ifndef RAD1O
|
|
usb_request_status_t usb_vendor_request_write_rffc5071(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
if (endpoint->setup.index < RFFC5071_NUM_REGS) {
|
|
rffc5071_reg_write(
|
|
&mixer,
|
|
endpoint->setup.index,
|
|
endpoint->setup.value);
|
|
usb_transfer_schedule_ack(endpoint->in);
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
return USB_REQUEST_STATUS_STALL;
|
|
} else {
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
}
|
|
|
|
usb_request_status_t usb_vendor_request_read_rffc5071(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
uint16_t value;
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
if (endpoint->setup.index < RFFC5071_NUM_REGS) {
|
|
value = rffc5071_reg_read(&mixer, endpoint->setup.index);
|
|
endpoint->buffer[0] = value & 0xff;
|
|
endpoint->buffer[1] = value >> 8;
|
|
usb_transfer_schedule_block(
|
|
endpoint->in,
|
|
&endpoint->buffer,
|
|
2,
|
|
NULL,
|
|
NULL);
|
|
usb_transfer_schedule_ack(endpoint->out);
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
return USB_REQUEST_STATUS_STALL;
|
|
} else {
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
usb_request_status_t usb_vendor_request_set_clkout_enable(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
si5351c_clkout_enable(&clock_gen, endpoint->setup.value);
|
|
usb_transfer_schedule_ack(endpoint->in);
|
|
}
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
|
|
usb_request_status_t usb_vendor_request_get_clkin_status(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
endpoint->buffer[0] = si5351c_clkin_signal_valid(&clock_gen);
|
|
usb_transfer_schedule_block(
|
|
endpoint->in,
|
|
&endpoint->buffer,
|
|
1,
|
|
NULL,
|
|
NULL);
|
|
usb_transfer_schedule_ack(endpoint->out);
|
|
}
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
|
|
usb_request_status_t usb_vendor_request_set_leds(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
set_leds(endpoint->setup.value);
|
|
usb_transfer_schedule_ack(endpoint->in);
|
|
}
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|
|
|
|
usb_request_status_t usb_vendor_request_user_config_set_bias_t_opts(
|
|
usb_endpoint_t* const endpoint,
|
|
const usb_transfer_stage_t stage)
|
|
{
|
|
if (stage == USB_TRANSFER_STAGE_SETUP) {
|
|
user_config_set_bias_t_opts(endpoint->setup.value);
|
|
usb_transfer_schedule_ack(endpoint->in);
|
|
}
|
|
return USB_REQUEST_STATUS_OK;
|
|
}
|