Introduction


Around 6 or 7 years ago I was at a flea market with a friend of mine: he was looking for an audio amp while I was looking for a tuner.
Guess what? We found a amplifier with tuner combo for a bargain price and thus I went home with the tuner while he went rocking his "new" amp.

The device is a Pioneer TX-950 - nothing special - but worked well so far.
A limitation is that the thing has no automatic tuning scan (i.e. press a button and forget till station is tuned).
Recently, I got the insane idea of "tuning" it (pardon the pun ;-) ) by modifying this aspect.

Initial research

The service manual can be found here: https://www.hifiengine.com/manual_library/pioneer/tx-950.shtml

The circuitry is based on a hybrid MCU which contains also the PLL and other tuner related stuff. Apparently, there are some pins to enable automatic scanning but unfortunately it seems that there is some missing circuitry around that to make it work.

Implementation idea

The user panel has two indicators: one for "Station Tuned" and one for "Station Tuned and Stereo".

So, I got the idea of simply monitoring the "Station Tuned" signal, the "Tune Up button" and "Tune Down button" and simulate keypresses by forcing the user buttons on demand.

In this way a simple logic can be applied to generate the button press, check the tuned signal and continue this process until a station is detected as tuned.

Implementation

I have implemented the idea by using the ATTiny85 which I had lying around in the junk box.

The service manual has been used to find and detect the actual "Station Tuned" signal on the board; buttons instead were pretty easy to trace.

Each signal is coupled via a 3.3k resistor. This is especially critical for keypresses as the microcontroller could bring the line down while the user keeps the button pressed.

The pictures are showing the attiny placement on the board, the 3 signals connections and Vcc / Gnd connections. The original circuitry works at +5V.

Here, the software code which is kept very simple and stupid (no interrupts and busy waiting):

/**
 * @file
 *
 * @brief Auto-scan of frequencies for Pioneer TX-950
 *
 * This program runs on ATTiny85 whose schematic is:
 *
 * N/C         | PB5 | C  | VCC | 4.8V TC9157P's Vcc
 * Button DW   | PB3 | H  | PB2 | N/C
 * Button UP   | PB3 | I  | PB1 | N/C
 * Ground      | GND | P  | PB0 | Tune Indicator driving circuit (R806)
 *
 * Connections to the signals on the board are done via 8 kOhm resistors,
 * just in case (especially as the buttons can be driven while the user
 * is pressing them).
 *
 * Test mode that performs a basic functional test is also available by
 * turning off the device, pressing UP and DW button at the same time
 * and then turning on the device. Pressing UP will go 10 frequencies
 * UP, vice-versa it goes 10 frequencies by pressing the DW button.
 *
 * The normal mode implements the actual feature.
 * If a UP or DOWN button is pressed for at least 300ms, then the auto-scan
 * begins:
 *  - simulate keypress to advance or decrease frequency
 *  - wait for 100ms checking if any button press is performed
 *    - if button press is performed: abort the auto scan
 *  - read the tuned signal: if present, then the receiver is tuned. auto-scan is done
 *    - if not present: continue scanning
 *
 * There is no timeout or end of spectrum checks. If the receiver is tuned at the limit,
 * it will keep tuning.
 *
 * Design notes
 * The design of the program is very simplistic and therefore redundant.
 * No timers, no interrupts: just a plain simple logic written in a couple of hours.
 */

#include <avr/io.h>
#include <avr/interrupt.h>
#include <util/delay.h>
#include <stdbool.h>
#include <stdint.h>

#define BTN_UP_PIN       (PB4)
#define BTN_DW_PIN       (PB3)
#define INPUT_TUNED_PIN  (PB0)

#define IS_TUNED           (bit_is_set(PINB, INPUT_TUNED_PIN) != false)
#define BTN_UP_PRESSED     (bit_is_set(PINB, BTN_UP_PIN) != false)
#define BTN_DW_PRESSED     (bit_is_set(PINB, BTN_DW_PIN) != false)
#define BTN_EITHER_PRESSED ((BTN_UP_PRESSED != false) || (BTN_DW_PRESSED != false))
#define BTN_UP_SET         (PORTB |= (_BV(BTN_UP_PIN)))
#define BTN_DW_SET         (PORTB |= (_BV(BTN_DW_PIN)))
#define BTN_UP_CLR         (PORTB &= ~(_BV(BTN_UP_PIN)))
#define BTN_DW_CLR         (PORTB &= ~(_BV(BTN_DW_PIN)))

bool test_mode_enabled = false;

/**
 * Enable the drive of the up or down as output.
 * @param up_or_down    TRUE: up FALSE: down
 */
void btn_drive(bool up_or_down)
{
    /* Initialize the pins as outputs */
    if (up_or_down == true)
    {
        PORTB &= ~_BV(BTN_UP_PIN); /* disable */
        DDRB  |= _BV(BTN_UP_PIN);  /* input      */
    }
    else
    {
        PORTB &= ~_BV(BTN_DW_PIN); /* disable */
        DDRB  |= _BV(BTN_DW_PIN);  /* input      */
    }
}

/**
 * Disable the drive of the button pins and configure
 * them as inputs to monitor their state.
 */
void btn_monitor(void)
{
    /* Initialize the pins as inputs */
    DDRB  &= ~_BV(BTN_UP_PIN); /* input      */
    PORTB &= ~_BV(BTN_UP_PIN); /* no pull-up */
    DDRB  &= ~_BV(BTN_DW_PIN); /* input      */
    PORTB &= ~_BV(BTN_DW_PIN); /* no pull-up */
}

/**
 * Create a pulse to simulate the user pressing the UP button
 */
void btn_press_up(void)
{
    btn_drive(true);
    BTN_UP_SET;
    _delay_ms(10);
    BTN_UP_CLR;
    btn_monitor();
}

/**
 * Create a pulse to simulate the user pressing the DOWN button
 */
void btn_press_dw(void)
{
    btn_drive(false);
    BTN_DW_SET;
    _delay_ms(10);
    BTN_DW_CLR;
    btn_monitor();
}

bool btn_up_is_hold(void)
{
    uint8_t i = 0;

    if (BTN_UP_PRESSED == true)
    { /* Pressed once, check if kept pressed */
        for (i = 0; i < 30; i++)
        {
            if (BTN_UP_PRESSED == false)
            {
                /* button has been released, not pressed for long */
                break;
            }
            _delay_ms(10);
        }
    }
    return (i>=30);
}

bool btn_dw_is_hold(void)
{
    uint8_t i = 0;

    if (BTN_DW_PRESSED == true)
    { /* Pressed once, check if kept pressed */
        for (i = 0; i < 30; i++)
        {
            if (BTN_DW_PRESSED == false)
            {
                /* button has been released, not pressed for long */
                break;
            }
            _delay_ms(10);
        }
    }
    return (i>=30);
}

/**
 * Wait 100ms for the tuned indication to be ready to be read.
 * Additionally, return if any of the buttons have been pressed in the
 * meantime.
 * @return TRUE: buttons pressed while waiting for the tuning
 *         FALSE: buttons not pressed while waiting for the tuning
 */
bool tune_wait_or_exit(void)
{
    uint8_t i = 0;
    bool exit_scan = false;

    for (i = 0; i < 100; i++)
    {
        if (BTN_EITHER_PRESSED == true)
        {
            exit_scan = true;
            while(BTN_EITHER_PRESSED);
        }
        _delay_ms(1);
    }

    return exit_scan;
}

/**
 * Test mode logic
 */
void app_test_mode(void)
{
    uint8_t i;

    if (BTN_UP_PRESSED)
    {
        while(BTN_UP_PRESSED);
        _delay_ms(500);

        for (i = 0; i < 10; i++)
        {
            btn_press_up();
            _delay_ms(1000);
        }
    }
    else if (BTN_DW_PRESSED)
    {
        while(BTN_DW_PRESSED);
        _delay_ms(500);

        for (i = 0; i < 10; i++)
        {
            btn_press_dw();
            _delay_ms(1000);
        }
    }
}

/**
 * Normal application logic
 */
void app_normal_mode(void)
{
    bool exit_scan = false;

    if (btn_up_is_hold() == true)
    {
        while(BTN_EITHER_PRESSED);
        do
        {
            btn_press_up();
            exit_scan = tune_wait_or_exit();
        } while (IS_TUNED == false && (exit_scan == false));
        _delay_ms(300);
    }
    else if (btn_dw_is_hold() == true)
    {
        while(BTN_EITHER_PRESSED);
        do
        {
            btn_press_dw();
            exit_scan = tune_wait_or_exit();
        } while (IS_TUNED == false && (exit_scan == false));
        _delay_ms(300);
    }
}

/**
 * Execute the main logic
 * @return not relevant as it runs on a microcontroller
 */
int main(void)
{
    /* Initialize the input pins */
    DDRB  &= ~_BV(INPUT_TUNED_PIN); /* input      */
    PORTB &= ~_BV(INPUT_TUNED_PIN); /* no pull-up */

    btn_monitor();

    /* Determine if user requires test mode to be enabled */
    test_mode_enabled = BTN_EITHER_PRESSED;

    if (test_mode_enabled == true)
    {
        _delay_ms(1000);
    }

    while(1)
    {
        if (test_mode_enabled == true)
        {
            app_test_mode();
        }
        else
        {
            app_normal_mode();
        }
    }
}

Final considerations

It really took me a couple of hours in total to complete the project.

I really had fun designing it ;-)