1ae7379baa
Test: I solemnly swear I tested this conflict resolution. Bug: 34250038 Change-Id: I4196a6d5a1365106643ac6c1548fd86faffead6f
231 lines
7.2 KiB
C
231 lines
7.2 KiB
C
/*
|
|
* Copyright (C) 2015 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#define LOG_TAG "radio_hal_tool"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
#include <log/log.h>
|
|
|
|
#include <hardware/hardware.h>
|
|
#include <hardware/radio.h>
|
|
#include <system/radio.h>
|
|
#include <system/radio_metadata.h>
|
|
|
|
// Global state variables.
|
|
const struct radio_tuner *hal_tuner = NULL;
|
|
|
|
void usage() {
|
|
printf("Usage: "
|
|
"./radio_hal_tool -l\n"
|
|
"-l: List properties global to the Radio.\n"
|
|
);
|
|
}
|
|
|
|
void list_all_properties(radio_hw_device_t *device) {
|
|
radio_hal_properties_t hal_properties;
|
|
device->get_properties(device, &hal_properties);
|
|
printf("Class: %d\n"
|
|
"Impl: %s\n"
|
|
"Tuners: %d\n"
|
|
"Bands: %d\n\n",
|
|
hal_properties.class_id, hal_properties.implementor, hal_properties.num_tuners,
|
|
hal_properties.num_bands);
|
|
|
|
uint32_t i;
|
|
for (i = 0; i < hal_properties.num_bands; i++) {
|
|
printf("Band Information\n"
|
|
"Type: %d\n"
|
|
"Connected: %d\n"
|
|
"Lower limit: %d\n"
|
|
"Upper limit: %d\n"
|
|
"Spacing: %d\n\n",
|
|
hal_properties.bands[i].type,
|
|
hal_properties.bands[i].antenna_connected,
|
|
hal_properties.bands[i].lower_limit,
|
|
hal_properties.bands[i].upper_limit,
|
|
hal_properties.bands[i].num_spacings);
|
|
}
|
|
}
|
|
|
|
void callback(radio_hal_event_t *event, void *cookie) {
|
|
printf("\nEvent detected\n"
|
|
"Type: %d\n", event->type);
|
|
}
|
|
|
|
void tune(radio_hw_device_t *device, int band_number) {
|
|
int ret;
|
|
radio_hal_properties_t hal_properties;
|
|
ret = device->get_properties(device, &hal_properties);
|
|
if (ret != 0) {
|
|
printf("Err: get_properties returned: %d\n", ret);
|
|
return;
|
|
}
|
|
|
|
if ((uint32_t) band_number >= hal_properties.num_bands) {
|
|
printf("Tuner number range should be: [0, %d]\n", hal_properties.num_bands);
|
|
}
|
|
printf("Setting band config as:\n"
|
|
"Type: %d\n"
|
|
"Connected: %d\n"
|
|
"Lower limit: %d\n"
|
|
"Upper limit: %d\n"
|
|
"Spacing: %d\n\n",
|
|
hal_properties.bands[band_number].type,
|
|
hal_properties.bands[band_number].antenna_connected,
|
|
hal_properties.bands[band_number].lower_limit,
|
|
hal_properties.bands[band_number].upper_limit,
|
|
hal_properties.bands[band_number].num_spacings);
|
|
int cookie = 0;
|
|
ret = device->open_tuner(
|
|
device, (const radio_hal_band_config_t *) (&(hal_properties.bands[band_number])), false,
|
|
callback, &cookie, &hal_tuner);
|
|
if (ret != 0) {
|
|
printf("Err: open_tuner returned: %d\n", ret);
|
|
return;
|
|
}
|
|
// It takes some time to apply the config which is currently set as 500ms in
|
|
// the stub implementation.
|
|
sleep(1);
|
|
|
|
// Stub tuner implementation will regard this magic channel as a valid channel to tune.
|
|
ret = hal_tuner->tune(hal_tuner, 87916, 0);
|
|
if (ret != 0) {
|
|
printf("Err: tune returned: %d\n", ret);
|
|
return;
|
|
}
|
|
// In the stub implementation it takes ~100ms to tune to the channel and the
|
|
// data is set rightafter.
|
|
sleep(1);
|
|
}
|
|
|
|
void get_tuner_metadata(radio_hw_device_t *device) {
|
|
// Get the metadata and print it.
|
|
radio_program_info_t info;
|
|
radio_metadata_allocate(&info.metadata, 87916, 0);
|
|
int ret;
|
|
ret = hal_tuner->get_program_information(hal_tuner, &info);
|
|
if (ret != 0) {
|
|
printf("Err: Get program info ret code: %d\n", ret);
|
|
return;
|
|
}
|
|
|
|
// Print the info.
|
|
printf("Metadata from the band\n");
|
|
int i;
|
|
for (i = 0; i < radio_metadata_get_count(info.metadata); i++) {
|
|
radio_metadata_key_t key;
|
|
radio_metadata_type_t type;
|
|
void *value;
|
|
size_t size;
|
|
|
|
radio_metadata_get_at_index(info.metadata, i, &key, &type, &value, &size);
|
|
|
|
printf("\nMetadata key: %d\n"
|
|
"Type: %d\n", key, type);
|
|
|
|
switch (type) {
|
|
case RADIO_METADATA_TYPE_INT:
|
|
printf("Int value: %d\n", *((int *) value));
|
|
break;
|
|
case RADIO_METADATA_TYPE_TEXT:
|
|
printf("Text value: %s\n", (char *) value);
|
|
break;
|
|
case RADIO_METADATA_TYPE_RAW:
|
|
printf("Raw value, skipping\n");
|
|
break;
|
|
case RADIO_METADATA_TYPE_CLOCK:
|
|
printf("UTC Epoch: %lld\n"
|
|
"UTC Offset: %d\n",
|
|
(long long)((radio_metadata_clock_t *) value)->utc_seconds_since_epoch,
|
|
((radio_metadata_clock_t *) value)->timezone_offset_in_minutes);
|
|
}
|
|
}
|
|
|
|
// Close the tuner when we are done.
|
|
ret = device->close_tuner(device, hal_tuner);
|
|
if (ret != 0) {
|
|
printf("Err: close_tuner returned: %d\n", ret);
|
|
}
|
|
}
|
|
|
|
int main(int argc, char** argv) {
|
|
// Open the radio module and just ask for the list of properties.
|
|
const hw_module_t *hw_module = NULL;
|
|
int rc;
|
|
rc = hw_get_module_by_class(RADIO_HARDWARE_MODULE_ID, RADIO_HARDWARE_MODULE_ID_FM, &hw_module);
|
|
if (rc != 0) {
|
|
printf("Cannot open the hw module. Does the HAL exist? %d\n", rc);
|
|
return -1;
|
|
}
|
|
|
|
radio_hw_device_t *dev;
|
|
rc = radio_hw_device_open(hw_module, &dev);
|
|
if (rc != 0) {
|
|
printf("Cannot open the device. Check that HAL implementation. %d\n", rc);
|
|
return -1;
|
|
}
|
|
printf("HAL Loaded!\n");
|
|
|
|
// If this is a list properties command - we check for -l command.
|
|
int list_properties = 0;
|
|
// Get metadata.
|
|
int get_metadata = 0;
|
|
// Tune. Takes a tuner number (see bands obtainaed by list_properties).
|
|
int should_tune = 0;
|
|
int band_number = -1;
|
|
|
|
int opt;
|
|
while ((opt = getopt(argc, argv, "lmt:")) != -1) {
|
|
switch (opt) {
|
|
case 'l':
|
|
list_properties = 1;
|
|
break;
|
|
case 't':
|
|
should_tune = 1;
|
|
band_number = atoi(optarg);
|
|
break;
|
|
case 'm':
|
|
get_metadata = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (list_properties) {
|
|
printf("Listing properties...\n");
|
|
list_all_properties(dev);
|
|
} else {
|
|
if (should_tune) {
|
|
if (band_number < 0) {
|
|
printf("Tuner number should be positive");
|
|
return -1;
|
|
}
|
|
printf("Tuning to a station...\n");
|
|
tune(dev, band_number);
|
|
}
|
|
if (get_metadata) {
|
|
if (!hal_tuner) {
|
|
printf("Please pass -t <band_number> to tune to a valid station to get metadata.");
|
|
exit(1);
|
|
}
|
|
get_tuner_metadata(dev);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|