Some days ago I posted an article about a RC-1 clone infrared remote trigger for Canon SLR cameras. You can find it in this link.
The design included a quick and dirty firmware developed using Energia, an Arduino like environment for the MPS430 MCUs. This firmware was enough for testing most of the hardware but it could be improved. Mainly:
- It didn't used the 32768 Hz watch crystal
- It didn't used the resistors included to measure the battery voltage
- It consumed 3.2mA in idle state so it was not efficient at all
Before talking about the code I should comment that there were a pair of errors in the schematic of the previous article. There are two changes that are indicated in blue in the following figure: The 22k resistors to measure the battery voltage were at incorrect pins. When soldering the components I changed the pins on the fly to ease the connections and I forget to update the schematic. In the same way the resistor and IR LED diode are swapped to ease, also, the connections.
![]() |
IR Trigger Schematic v1.4 |
GCC Firmware
The new firmware was developed on the Portable MSP320 Environment I set up some days ago. The Texas Instruments sponsored Red Hat MSP430 GCC I used is a mixed bag. On one hand it provides an easy setup and an up to date selection of MCUs but on the other hand is not fully compatible with previous MSP430 code I developed for the MSP430 GCC at Sourceforge.
The firmware code is distributed in three files:
This file includes some macros to ease the acces to bit fields on the MCU hardware registers.
I developed this file long time ago but I needed to do some modifications for this project due to the change of toolchain. Due to that, it is not guaranteed that all this file works in a particular toolchain.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/********************************************* | |
Masks for MSP430 register bit fields | |
(c) VJS | |
Version 23-11-2014 | |
History: | |
28-06-2011 : First version | |
17-04-2012 : Some mask definition changes | |
14-11-2014 : Now all is in english | |
23-11-2014 : CLEAR_FLAG alias and ALL_BITS | |
**********************************************/ | |
/********************************************* | |
Example for a 8 bit register REG with | |
1 bit field BIT | |
2 bit field F0,F1 | |
REG|=BIT To set BIT | |
REG&=~BIT To clear BIT | |
For F0,F1 a mask is defined: | |
F01_MASK = (0xFF-3*F0) | |
REG&=F01_MASK To clear field | |
REG=(REG&F01_MASK)+F0 To set F0 and not F1 | |
REG=(REG&F01_MASK)+F1 To set F1 and not F0 | |
REG=(REG&F01_MASK)+F0+F1 To set F0 and F1 | |
REF=(REG&F01_MASK)+(F0*value) To set value 0..3 | |
***********************************************/ | |
/*************** BASIC DEFINITIONS ***************** | |
This definitions define the following macros: | |
SET_FLAG(REGISTER,FLAG) Set a 1 bit flag | |
RESET_FLAG(REGISTER,FLAG) Reset a 1 bit flag | |
CLEAR_FLAG(REGISTER,FLAG) Reset a 1 bit flag | |
CLEAR_FIELD(REGISTER,MASK) Clear a multibit field | |
SET_FIELD(REGISTER,MASK,VALUE) | |
Set a multibit field (value must be consistent with mask) | |
SET_FIELD_WOFFSET(REGISTER,MASK,VALUE,OFFSET) | |
Set a multibit field adding an offset | |
Field name references the last bit in the field, not the first one | |
** In the future macros can be changed to use BIS and BIC | |
***************************************************/ | |
#define SET_FLAG(REGISTER,FLAG) REGISTER|=(FLAG) | |
#define RESET_FLAG(REGISTER,FLAG) REGISTER&=~(FLAG) | |
#define CLEAR_FIELD(REGISTER,MASK) REGISTER&=(MASK) | |
#define SET_FIELD(REGISTER,MASK,VALUE) REGISTER=(REGISTER&(MASK))+VALUE | |
#define SET_FIELD_WOFFSET(REGISTER,MASK,VALUE,OFFSET) REGISTER=(REGISTER&(MASK))+((VALUE)*(OFFSET)) | |
#define GET_FIELD(REGISTER,MASK) (REGISTER&(MASK)) | |
#define CLEAR_FLAG(REGISTER,FLAG) RESET_FLAG(REGISTER,FLAG) | |
#define ALL_BITS (BIT0|BIT1|BIT2|BIT3|BIT4|BIT5|BIT6|BIT7) | |
/**************** ADC 10 **********************/ | |
#ifdef __MSP430_HAS_ADC10__ | |
/* ADC10CTL0 @ 0x01B0 ADC10 Control 0 */ | |
#define ADC10SHT_OFFS (0x800u) | |
#define ADC10SHT_MASK (0xFFFF-3*ADC10SHT_OFFS) /* 2 bit mask */ | |
#define ADC10SREF_OFFS (0x2000u) | |
#define ADC10SREF_MASK (0xFFFF-7*ADC10SREF_OFFS) /* 3 bit mask */ | |
/* ADC10CTL1 @ 0x01B2 ADC10 Control 1 */ | |
#define ADC10CONSEQ_OFFS (2u) | |
#define ADC10CONSEQ_MASK (0xFFFF-3*ADC10CONSEQ_OFFS) /* 2 bit mask */ | |
#define ADC10SSEL_OFFS (8u) | |
#define ADC10SSEL_MASL (0xFFFF-3*ADC10SSEL_OFFS) /* 2 bit mask */ | |
#define ADC10DIV_OFFS (0x20u) | |
#define ADC10DIV_MASK (0xFFFF-7*ADC10DIV_OFFS) /* 3 bit mask */ | |
#define SHS_OFFS (0x400u) | |
#define SHS_MASK (0xFFFF-3*SHS_OFFS) /* 2 bit mask */ | |
#define INCH_OFFS (0x1000u) | |
#define INCH_MASK (0xFFFFu-0xF*INCH_OFFS) /* 4 bit mask */ | |
#endif | |
/********* BASIC CLOCK MODULE ****************/ | |
#ifdef __MSP430_HAS_BC2__ | |
/* DCOCTL @ 0x0056 DCO Clock Frequency Control */ | |
#define MOD_OFFS (MOD0) | |
#define MOD_MASK (0xFF-0x1F*MOD0) /* 5 bit mask */ | |
#define DCO_OFFS (DCO0) | |
#define DCO_MASK (0xFF-7*DCO0) /* 3 bit mask */ | |
/* BCSCTL1 @ 0x0057 Basic Clock System Control 1 */ | |
#define RSEL_OFFS (RSEL0) | |
#define RSEL_MASK (0xFF-0xF*RSEL0) /* 4 bit mask */ | |
#define DIVA_OFFS (DIVA0) | |
#define DIVA_MASK (0xFF-0x3*DIVA0) /* 2 bit mask */ | |
/* BCSCTL2 @ 0x0058 Basic Clock System Control 2 */ | |
#define DIVS_OFFS (DIVS0) | |
#define DIVS_MASK (0xFF-0x3*DIVS0) /* 2 bit mask */ | |
#define DIVM_OFFS (DIVM0) | |
#define DIVM_MASK (0xFF-0x3*DIVM0) /* 2 bit mask */ | |
#define SELM_OFFS (SELM0) | |
#define SELM_MASK (0xFF-0x3*SELM0) /* 2 bit mask */ | |
/* BCSCTL3 @ 0x0053 Basic Clock System Control 3 */ | |
#define XCAP_OFFS (XCAP0) | |
#define XCAP_MASK (~(XCAP0|XCAP1)) /* 2 bit mask */ | |
#define LFXT1S_OFFS (LFXT1S0) | |
#define LFXT1S_MASK (~(LFXT1S0|LFXT1S1)) /* 2 bit mask */ | |
#define XT2S_OFFS (XT2S0) | |
#define XT2S_MASK (0xFF-0x3*XT2S0) /* 2 bit mask */ | |
#endif | |
/******** FLASH MEMORY *************************/ | |
#ifdef __MSP430_HAS_FLASH2__ | |
/* FCTL2 @ 0x012A FLASH Control 2 */ | |
#define FN_OFFS (FN0) | |
#define FN_MASK (0xFF-0x3F*FN0) /* 6 bit mask */ | |
#define FSSEL_OFFS (FSSEL0) | |
#define FSSEL_MASK (0xFF-0x3*FSSEL0) /* 2 bit mask */ | |
#endif | |
/******* DIGITAL I/O PORTS 1/2 *****************/ | |
/* No definitions are needed here */ | |
/********** TIMER A2 ***************************/ | |
#ifdef __MSP430_HAS_TA2__ | |
#define __MSP430_HAS_TA__ | |
#endif | |
#ifdef __MSP430_HAS_TA3__ | |
#define __MSP430_HAS_TA__ | |
#endif | |
#ifdef __MSP430_HAS_TA__ | |
/* TAIV @ 0x012E Timer A Interrupt Vector Word (Read Only) */ | |
#define IRQ_OFFS (0x0x2) | |
#define IRQ_MASK (0xFFFF-7*IRQ_OFFS) /* 3 bit mask */ | |
/* Mask not normally needed as it is read only !!! */ | |
/* TACTL @ 0x0160 Timer A Control */ | |
#define TAMC_OFFS (0x10u) | |
#define TAMC_MASK (0xFFFF-3*TAMC_OFFS) /* 2 bit mask */ | |
#define TAID_OFFS (0x40u) | |
#define TAID_MASK (0xFFFF-3*TAID_OFFS) /* 2 bit mask */ | |
#define TASSEL_OFFS (0x100u) | |
#define TASSEL_MASK (0xFFFF-3*TASSEL_OFFS) /* 2 bit mask */ | |
/* TACCTL0 @ 0x0162 Timer A Capture/Compare Control 0 */ | |
#define OUTMOD_OFFS (0x20u) | |
#define OUTMOD_MASK (0xFFFF-0x7*OUTMOD_OFFS) /* 3 bit mask */ | |
#define CCIS_OFFS (0x1000u) | |
#define CCIS_MASK (0xFFFF-0x3*CCIS_OFFS) /* 2 bit mask */ | |
#define CM_OFFS (0x4000u) | |
#define CM_MASK (0xFFFF-0x3*CM_OFFS) /* 2 bit mask */ | |
/* TACCTL1 @ 0x0164 Timer A Capture/Compare Control 1 */ | |
/* Masks are the same as in TACCTL0 */ | |
/* Aditional Definitions */ | |
#define TAIV_NONE 0x00 | |
#define TAIV_TACCR1 0x02 | |
#define TAIV_TACCR2 0x04 | |
#define TAIV_TAIFG 0x0A | |
#endif | |
/************** USI ****************************/ | |
#ifdef __MSP430_HAS_USI__ | |
/* USICKCTL @ 0x007A USI Clock Control Register */ | |
#define USISSEL_OFFS (USISSEL0) | |
#define USISSEL_MASK (0xFF-7*USISSEL0) /* 3 bit mask */ | |
#define USIDIV_OFFS (USIDIV0) | |
#define USIDIV_MASK (0xFF-7*USIDIV0) /* 3 bit mask */ | |
/* USICNT @ 0x007B USI Bit Counter Register */ | |
#define USICNT_OFFS (USICNT0) | |
#define USICNT_MASK (0xFF-0x1F*USICNT0) /* 5 bit mask */ | |
#endif | |
/************** WATCHDOG **********************/ | |
/* Mask cannot be used as it is protected by a password */ | |
/************** COMPARATOR A *****************/ | |
#ifdef __MSP430_HAS_COMPA__ | |
/* CACTL1 @ 0x0059 Comparator A Control 1 */ | |
#define CAREF_OFFS (CAREF0) | |
#define CAREF_MASK (0xFF-0x3*CAREF0) /* 2 bit mask */ | |
/* CACTL2 @ 0x005A Comparator A Control 2 */ | |
#define P2CA_OFFS (P2CA0) | |
#define P2CA_MASK (0xFF-0x3*P2CA0) /* 2 bit mask */ | |
/* Definitions for missing signals in .h */ | |
#define P2CA2 BIT4 | |
#define P2CA3 BIT5 | |
#define P2CA4 BIT6 | |
#define CASHORT BIT7 | |
#define SELP_MASK (0xFF-P2CA0-P2CA4) | |
#define SELP_NONE (0x00) | |
#define SELP_CA0 (P2CA0) | |
#define SELP_CA1 (P2CA4) | |
#define SELP_CA2 (P2CA0+P2CA4) | |
#define SELN_MASK (0xFF-7*P2CA1) | |
#define SELN_NONE (0x00) | |
#define SELN_CA1 (1*P2Ca1) | |
#define SELN_CA2 (2*P2Ca1) | |
#define SELN_CA3 (3*P2Ca1) | |
#define SELN_CA4 (4*P2Ca1) | |
#define SELN_CA5 (5*P2Ca1) | |
#define SELN_CA6 (6*P2Ca1) | |
#define SELN_CA7 (7*P2Ca1) | |
#endif |
This is the main file for the firmware. It practically contains all the code. It also includes some test code not used in the final firmware but that was left behind.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/***************/ | |
/* IR TRigger */ | |
/***************/ | |
// RC-1 Equivalent trigger for Canon EOS | |
// Frequency: 32700 Hz (29800 to 35500 Hz) | |
// Pulses/Burst: 16 (9 to 22) | |
// Two burst start delay: Immediate 7.82ms | |
// Delayed (2s) 5.86ms | |
// History | |
// 25/11/2014 First version (v1.0) | |
// Use low power LPM4 mode | |
// Energia first firmware had 3.2mA idle current | |
// This firmware has a 0.1uA idle current | |
// This is just the typical value t 25ºC reported on the datasheet | |
// TODO : The first burst is not always correct | |
// Perhaps we can go to LPM3 and then to LPM4 after several seconds | |
// Use IR LED TSHF5210 | |
#include <msp430.h> | |
#include "io430masks.h" | |
// Main configuration definitions | |
//#define RELEASE // Release version (Don't include test functions) | |
// Comment it during product debugging | |
// Hardware definitions | |
// PORT 1 USAGE | |
#define LED BIT0 // Active High Red LED at pin 2 (P1.0) | |
#define SW1 BIT3 // Button between pin 5 (P1.3) and GND | |
#define SW2 BIT5 // Button between pin 7 (P1.5) and GND | |
#define IRa BIT4 // Active High IR LED at pin 6 (P1.4) | |
// P1.4 can be used as SMCLK output | |
//#define RD_H BIT7 // Upper end of resistor divider at pin 15 (P1.7) | |
//#define RD_M BIT6 // Middle point of resistor divider at pin 14 (P1.6) (A6) | |
#define RD_M BIT7 // Middle point of resistor divider at pin 15 (P1.7) (A7) | |
#define UNUSED_P1 ( BIT1 | BIT2 | BIT6 ) // Unused P1 lines | |
// PORT 2 USAGE | |
#define IRb BIT1 // Active High IR LED also on pin 9 (P2.1) | |
// P2.1 is Timer1_A Out1 output in compare mode | |
#define RD_H BIT3 // Upper end of resistor divider at pin 11 (P2.3) | |
// P2.6 and P2.7 are used by the 32768 Xtal | |
#define UNUSED_P2 ( BIT0 | BIT2 | BIT4 | BIT5 ) // Unused P2 lines | |
// PORT 3 USAGE | |
// | |
// There is no available Port3 in the 20 pin DIP package | |
// however we must disable port3 to obtain low current LPM4 | |
#define UNUSED_P3 ( ALL_BITS ) | |
/************* STATUS MACHINE DEFINITIONS *******************/ | |
#define ST_NONE 0 // Nothing to do | |
#define ST_NFB 1 // Normal trigger first burst ( 16 cycles) | |
#define ST_ND 2 // Normal trigger delay between bursts (240 cycles) | |
#define ST_SB 3 // Normal/Delayed trigger second burst ( 16 cycles) | |
#define ST_DFB 5 // Delayed trigger first burst ( 16 cycles) | |
#define ST_DD 6 // Delayed trigger delay between bursts (176 cycles) | |
/**************** VARIABLES *********************************/ | |
unsigned int fclk_MHz=1; // Clock frequency in MHz | |
// Used by the delay function | |
volatile int buttons=0; // Buttons to be processed | |
// Status variable for timer | |
// | |
// 0 : Nothing to do | |
// | |
// Immediate trigger codes | |
// 1 : First burst | |
// 2 : Wait for second burst | |
// 3 : Second burst | |
// | |
// Delayed trigger codes | |
// 5 : First burst | |
// 6 : Wait for second burst | |
// 7 : Second burst | |
// | |
volatile int status=ST_NONE; | |
/************** FUNCTION PROTOTYPES *************************/ | |
int checkVdd(void); | |
/**************** FUNCTIONS *********************************/ | |
// Delay of about len ms | |
void delay(unsigned int len) | |
{ | |
if (!len) return; // Nothing to do if len=0 | |
// Set number of cycles depending on clk frequency | |
len*=fclk_MHz; | |
do | |
{ | |
// 1000 cycles delay | |
__delay_cycles (1000); | |
len--; // Decrease counter | |
} | |
while(len); | |
} | |
// Sends several blinks to the RED LED | |
// Parameters: | |
// n : Number of blinks | |
// length : ON, OFF Duration in ms | |
void doBlinks(unsigned int n,unsigned int length) | |
{ | |
do | |
{ | |
SET_FLAG(P1OUT,LED); // Turn on led | |
delay(length); | |
CLEAR_FLAG(P1OUT,LED); // Turn off led | |
delay(length); | |
} | |
while(--n); | |
} | |
/**************** TEST FUNCTIONS ***************************/ | |
// These functions are used only during the development | |
// They can be eliminated in the release product | |
#ifndef RELEASE | |
// Changes blink frequency depending on SW1 and SW2 | |
// This is a test function. Never returns | |
void blinkTest(void) | |
{ | |
while (1) | |
{ | |
SET_FLAG(P1OUT,LED); // Turn on led | |
if (P1IN&SW1) | |
delay(1000); // Time ON 1s if SW1 not pressed | |
else | |
delay(200); // Time ON 0.2s if SW1 pressed | |
CLEAR_FLAG(P1OUT,LED); // Turn off led | |
if (P1IN&SW2) | |
delay(1000); // Time OFF 1s if SW2 not pressed | |
else | |
delay(200); // Time OFF 0.2s if SW2 pressed | |
} | |
} | |
// Test outputting SMCLK at P1.4 (IRa) | |
// This is a test function. Never returns | |
void smclkTest(void) | |
{ | |
// Configurations to set SMCLK | |
SET_FLAG(P1DIR,IRa); | |
SET_FLAG(P1SEL,IRa); | |
} | |
#endif /* RELEASE */ | |
/********** CAMERA TRIGGER FUNCTIONS **********************/ | |
// Normal camera trigger switch SW1 is pressed | |
// Two burst of 16 ACLK cycles | |
// Starts separated 7.82ms (256 cycles) | |
void normalTrigger(void) | |
{ | |
// doBlinks(2,200); // Test that we enter OK | |
SET_FLAG(P1SEL,IRa); // Activate burst | |
SET_FLAG(TACTL,TACLR); // Clear the counter | |
TACCR0=16; // First interrupt set at 16 cycles | |
status=ST_NFB; // This is first burst of normal trigger | |
SET_FIELD(TACTL,TAMC_MASK,MC_2); // Set mode to Up Continuous | |
delay(100); | |
if (checkVdd()) | |
SET_FLAG(P1OUT,LED); // Turn on led | |
delay(400); | |
CLEAR_FLAG(P1OUT,LED); // Turn off led | |
} | |
// Delayed camera trigger switch SW2 is pressed | |
// Two burst of 16 ACLK cycles | |
// Starts separated 5.86ms (192 cycles) | |
void delayedTrigger(void) | |
{ | |
// doBlinks(4,200); // Test that we enter OK | |
SET_FLAG(P1OUT,LED); // Turn on led | |
SET_FLAG(P1SEL,IRa); // Activate burst | |
SET_FLAG(TACTL,TACLR); // Clear the counter | |
TACCR0=16; // First interrupt set at 16 cycles | |
status=ST_DFB; // This is first burst of delayed trigger | |
SET_FIELD(TACTL,TAMC_MASK,MC_2); // Set mode to Up Continuous | |
delay(100); | |
if (checkVdd()) | |
SET_FLAG(P1OUT,LED); // Turn on led | |
delay(400); | |
CLEAR_FLAG(P1OUT,LED); // Turn off led | |
} | |
/*************** ADC FUNCTIONS ****************************/ | |
unsigned int value; | |
// Reads the indicated channel on the ADC10 | |
// Uses an internal reference of 1,5V | |
unsigned int readADC10(int channel) | |
{ | |
unsigned int val; | |
//ADC10CTL0 | |
// Activation of the ADC10 core | |
//SET_FLAG(ADC10CTL0,ADC10ON); | |
// Use default internal clock source | |
// Use default divider | |
// Enable the internal 2.5V reference | |
//SET_FLAG(ADC10CTL0,REFON); | |
// Set the maximum S/H | |
//SET_FIELD(ADC10CTL0,ADC10SHT_MASK,ADC10SHT_3); | |
// Use VR+ = VREF+ VR- = AVSS | |
// SET_FIELD(ADC10CTL0,ADC10SREF_MASK,SREF_1); | |
// More compact solution | |
// Default triggering with ADC10SC | |
// Default single channel single conversion | |
ADC10CTL0=ADC10ON|REFON|ADC10SHT_3|SREF_1; | |
// Enable the channel | |
ADC10AE0=(1<<channel); | |
// Set the input | |
ADC10CTL1=(((unsigned int)channel)*INCH_OFFS)|ADC10DIV_7; | |
delay(10); | |
// Set the conversion | |
ADC10CTL0|=(ENC|ADC10SC); | |
// Wait for conversion termination | |
while (ADC10CTL1&ADC10BUSY) {}; | |
// ENC must be cleared before accessing | |
// the rest of the flags on ADC10CTL0 | |
// Failing to do that can leave some peripherals | |
// on ADC activated and ruin low power in LPM4 mode | |
RESET_FLAG(ADC10CTL0,ENC); | |
// Get the converted value | |
val=ADC10MEM; | |
// Disable the channel | |
ADC10AE0=0; | |
// Disable OSC ADC core and Reference | |
CLEAR_FLAG(ADC10CTL0,ADC10SC|ADC10ON|REFON); | |
//ADC10CTL0=0; | |
//ADC10CTL1=0; | |
return val; | |
} | |
// Check the Vdd voltage | |
// TODO Test that it works ok | |
// Returns 0 if below 2.6V | |
// Returns 1 otherwise | |
int checkVdd(void) | |
{ | |
// Activate the resistor divider | |
SET_FLAG(P2OUT,RD_H); | |
// Wait 10ms to settle | |
delay(10); | |
// Read ADC at center point | |
value=readADC10(7); | |
value=readADC10(7); | |
// Deactivate the divider | |
CLEAR_FLAG(P2OUT,RD_H); | |
// We consider the battery nearly depleted when | |
// the voltage falls below 2.6V | |
// That means 1024*(2.6/2/1.5) = 890 | |
if (value<890) return 0; | |
return 1; | |
} | |
/*************** MAIN FUNCTION ****************************/ | |
int main(void) | |
{ | |
WDTCTL = WDTPW | WDTHOLD; // Stop watchdog timer | |
P1DIR |= LED | IRa; // Output mode pins | |
P1OUT = 0; // Guarantee that LED and IR are off | |
P1REN |= SW1 | SW2; // Activate pull-up/pull-down | |
P1OUT |= SW1 | SW2; // Set to pull-up | |
// Invalidate unused pins using pull-downs | |
P1REN |= UNUSED_P1; | |
CLEAR_FLAG(P1OUT,UNUSED_P1); | |
CLEAR_FLAG(P1DIR,UNUSED_P1); | |
P2REN |= UNUSED_P2; | |
CLEAR_FLAG(P2OUT,UNUSED_P2); | |
CLEAR_FLAG(P2DIR,UNUSED_P2); | |
P3REN |= UNUSED_P3; | |
CLEAR_FLAG(P3OUT,UNUSED_P3); | |
CLEAR_FLAG(P3DIR,UNUSED_P3); | |
// Set resistor divider high node | |
P2DIR |= RD_H; | |
CLEAR_FLAG(P2OUT,RD_H); | |
// Set 32768kHz Xtal capacitance to 12.5pF | |
SET_FIELD(BCSCTL3,XCAP_MASK,XCAP_3); | |
// Select 32768kHz LFXTCLK for SMCLK | |
SET_FLAG(BCSCTL2,SELS); | |
// Load calibrated data for 16MHz clock | |
DCOCTL=CALDCO_16MHZ; // DCOCTL Calibration Data for 16MHz | |
BCSCTL1=CALBC1_16MHZ; // BCSCTL1 Calibration Data for 16MHz | |
fclk_MHz=16; // Set clk data for timings | |
// Wait 1 second | |
delay(1000); | |
// Interrupt configuration for SW1 and SW2 | |
P1IES |= SW1|SW2; // Sets pins SW1 and SW2 port 1 to falling edge | |
P1IE |= SW1|SW2; // Enables interrupts on SW1 and SW2 | |
P1IFG=0; // Clear interrupt flags | |
// Configure Timer0A | |
SET_FIELD(TACTL,TASSEL_MASK,TASSEL_1); // Set clock to ACLK (32768Hz) | |
SET_FIELD(TACTL,TAID_MASK,ID_0); // Set clock divider to :1 | |
SET_FIELD(TACTL,TAMC_MASK,MC_0); // Set mode to STOP | |
SET_FLAG(TACTL,TACLR); // Clear the counter | |
// Configure Capture block 0 | |
CLEAR_FLAG(TACCTL0,CAP); // Mode set to compare | |
TACCR0=16; // First interrupt set at 16 cycles | |
SET_FLAG(TACCTL0,CCIE); // Enable interrupt | |
// Blink test | |
// At 25/11/2014 it works OK, we don't need it anymore | |
// blinkTest(); | |
// SMCLK Test | |
// At 25/11/2014 it works OK, we get 32768Hz on the IR LED | |
// smclkTest(); | |
// Signal the end of the configuration | |
doBlinks(2,200); // 2 blinks of 200ms | |
_enable_interrupts(); // Enable interrupts | |
// Infinite loop | |
while (1) | |
{ | |
// Check if there is any SW to respond to | |
if (!buttons) | |
{ | |
LPM4; // Go to sleep if there are no buttons | |
delay(200); // Little delay going out of sleep | |
} | |
if (buttons&SW1) // Normal camera trigger | |
{ | |
normalTrigger(); | |
buttons=0; | |
} | |
if (buttons&SW2) // Delayed camera trigger | |
{ | |
delayedTrigger(); | |
buttons=0; | |
} | |
}; | |
} | |
/*********************** GPIO INTERRUPT **************************/ | |
// The port 1 is associated with vectors PORT1_VECTOR | |
// The RSI must test the different bits | |
// It seems that the interrupt() macro in iomacros.h is broken | |
// #define __interrupt(vec) __attribute__((__interrupt__(vec))) | |
// We can redefine a new macro: | |
#define interruptNew(vec) __attribute__((interrupt(vec))) | |
// PORT 1 RSI | |
void interruptNew(PORT1_VECTOR) PORT1_ISR (void) | |
{ | |
// Don't do anything if we have not processed last order | |
if (!buttons) | |
{ | |
buttons|=P1IFG&(SW1|SW2); | |
// Exit LPM4 mode on exit | |
LPM4_EXIT; | |
} | |
// Clear the flags | |
P1IFG=0; | |
} | |
/************** TIMER0A COMPARE 0 INTERRUPT ********************/ | |
void interruptNew(TIMER0_A0_VECTOR) TA0Capture0_ISR(void) | |
{ | |
switch (status) | |
{ | |
case ST_NONE: | |
// Nothing to do | |
break; | |
case ST_NFB: | |
// We were in the first burst | |
CLEAR_FLAG(P1SEL,IRa); // Deactivate burst | |
TACCR0+=240; // 240 cycles to next change | |
status=ST_ND; // Now in Delay between Bursts | |
break; | |
case ST_ND: | |
// We were in the space between bursts | |
SET_FLAG(P1SEL,IRa); // Activate burst | |
TACCR0+=16; // 16 cycles to next change | |
status=ST_SB; // Now in Second Burst | |
break; | |
case ST_SB: | |
// We were in the second burst | |
CLEAR_FLAG(P1SEL,IRa); // Deactivate burst | |
SET_FIELD(TACTL,TAMC_MASK,MC_0); // Set mode to STOP | |
status=ST_NONE; // Now in Idle mode | |
break; | |
case ST_DFB: | |
// We were in the first burst | |
CLEAR_FLAG(P1SEL,IRa); // Deactivate burst | |
TACCR0+=176; // 176 cycles to next change | |
status=ST_DD; // Now in Delay between bursts | |
break; | |
case ST_DD: | |
// We were in the space between bursts | |
SET_FLAG(P1SEL,IRa); // Activate burst | |
TACCR0+=16; // 16 cycles to next change | |
status=ST_SB; // Now in Second Burst | |
break; | |
default: | |
// Default case if anything goes wrong | |
SET_FIELD(TACTL,TAMC_MASK,MC_0); // Set mode to STOP | |
CLEAR_FLAG(P1SEL,IRa); // Deactivate burst | |
status=ST_NONE; // Go to a valid state | |
break; | |
} | |
} |
This is the makefile that coordinates the build of all the firmware.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# IR Trigger Makefile | |
# Folders to include in the path | |
# M:\MCU\msp430_Luna\UnxUtils\usr\local\wbin | |
# M:\MCU\msp430_Luna\msp430-gcc\bin | |
NAME = IR_Trigger | |
OBJECTS = main.o | |
MCU = msp430g2553 | |
#Compiler | |
GCC_DIR = ../../msp430-gcc | |
CC = msp430-elf-gcc | |
# It seems that the GCC compiler don't know that | |
# the selected MCU has no hardware multiplier | |
# so we need to indicate that in the compiler flags | |
# Development version | |
#CFLAGS = -I $(GCC_DIR)/include -mmcu=$(MCU) -mhwmult=none -O0 -g | |
# Release version | |
CFLAGS = -I $(GCC_DIR)/include -mmcu=$(MCU) -mhwmult=none -O2 | |
#Linker | |
LFLAGS = -L $(GCC_DIR)/include -Wl,-Map=$(NAME).map | |
#Object Dump | |
#OD = ${GCC_DIR}/bin/msp430-elf-objdump | |
OD = msp430-elf-objdump | |
.PHONY: all clean | |
all: ${NAME}.elf ${NAME}.lst | |
${NAME}.elf: ${OBJECTS} | |
$(CC) $(CFLAGS) $(LFLAGS) $? -o $(NAME).elf | |
${NAME}.lst: ${NAME}.elf | |
${OD} -dSt $^ > $@ | |
clean: | |
rm -f ${NAME}.elf ${NAME}.lst ${OBJECTS} | |
#project dependencies | |
main.o: main.c io430masks.h Makefile |
The code starts stopping the Watchdog and configuring the I/O ports. In particular it is very important to leave guarantee than there is no I/O floating as they will ruin the low power figures. In order to tie the unused lines to GND, the pull-downs are enabled on these lines.
Next, the 32768 Hz ACLK clock is configured and set as SMCLK. Then the main DCO clock is configured to operate at 16MHz using the Flash stored factory calibrated values.
After that, the push buttons SW1 and SW2 are configured to generate an interrupt when pressed. The configuration ends with the Timer0A that is connected to the 32768 Hz ACLK as input and associated with a capture interrupt RSI.
At the end of the initialization the red LED gives two blinks to indicate that the system is ready to operate.
After the initialization there is an infinite event loop. There are two possible events, each one associated to a function. If there is no event to process, the system enters in deep sleep LPM4 mode.
The Port 1 RSI associated to the pushbuttons generates the two possible events and gets out of LPM4 mode if needed so that the main event loop can process them.
As it was indicated there are two kinds of events. The SW1 button generates Normal Trigger Events that sends through the IR LED the signal that makes the camera shoot immediately. The SW2 button generates the Delayed Trigger Event that sends a similar signal that makes the camera shoot with a 2 seconds delay.
The signals associated to each event are composed on two 32768 Hz 16 cycles square wave bursts separated with some delay. That delay is used by the camera to identify both signals. The Normal Trigger signal has both bursts starts separated 7.82ms whereas the Delayed Trigger signal uses a 5.86ms separation.
The signals associated to those events are generated with an state machine associated to the Timer. An ISR execution associated to the Capture channel 0 of the Timer 0 provides the change from one state to the following one. The entry point to the state machine for the Normal Trigger Event is the Normal First Burst (NFB) state that connects the ACLK to the P1.4 pin during 16 cycles. After the state times out, the following Normal Delay (ND) state waits during 240 clock cycles. Then the Second Burst (SB) state outputs the second burst of 16 clock cycles to the IR LED. That ends the state machine for the Normal Trigger.
There is a sencod path in the state machine for the Delayed Trigger signal that starts with the Delayed First Burst (DFB) state. The only difference is this path is the second Delayed Delay (DD) state that is 176 cycles long instead of the 240 cycles associated to the ND state.
To provide some security against random errors, any undefined state goes to the NONE estate that stops the machine and also the associated timer.
![]() |
Timer State Machine |
The use of an state machine is an efficient way to sequence the operations associated to the generation of the IR signal. At the start of the first state NFB, the SMCLK clock is sent to the P1.4 pin, the Compare register TACCR0 is set to 16 cycles in the future and the timer counter TAR is started in countinuous up mode.
![]() |
Timer 0 TAR evolution and States |
When the TAR counter reaches the TACCR0 value, an interrupt is generated. That changes the state from NFB to ND. The P1.4 pin is disconnected from the SMCLK clock and the TACCR0 Compare register is set 176 cycles in the future. When TAR reaches the 256 cycle mark a new interrupt is generated giving start to the SB state. A new burst is generated and TACCR0 is set 16 cycles in the future. Finally, at the 272 cycle mark, we enter in the NONE state where the burst ends and the TAR timer counter is stopped.
Measurements
I have made some measurements to test the operation of the system. The following image shows the voltage at the IR LED diode during the first burst. We can see that there are 16 pulses.
![]() |
Burst signal at the IR LED |
To measure the clock frequency we can zoom in the burst pulses at the LED. We can see from it that the period is 30.6us so that the frequency is about 32680 Hz. As the start of the signal is not synchronized with the clock, the first pulse is not equal to the rest. In practice this is not a problem but, if it were, we could add another state in the machine to synchronize the burst start.
![]() |
Burst pulse detail |
The following figure shows the IR LED voltage for the full signal in normal non delayed camera trigger mode. We can measure a 7.81ms delay time between the start of both bursts. This is just the desired value.
![]() |
Normal Trigger Full signal |
![]() |
Delayed Trigger Full signal |
To test the real current on the diode we can measure the voltage on the current limiting 6.8Ohm resistor that is in series with the LED. We can see that the peak pulse voltage goes from 560mV at the start of the burst to about 470mV at its end. That means that the peak current goes from 82mA to 69mA. We planned for 100mA current and we could lower the current limiting resistor to increase the current, but in the end the CR2032 battery that powers the system could produce system resets due to the fall in the supply during the bursts.
![]() |
Voltage at 6.8 Ohm resistor |
Idle current
One of the most important drawbacks in the Energia firmware was the 3.2mA idle current. Using the GCC firmware we can use the low power LPM4 mode to significantly reduce this current.From the information provided on the MSP430G2553 we know that the LPM4 current can be as low as 0.1uA.
![]() |
Low Power figures in the MSP430G2553 |
Using this method I obtained in practice the datasheet indicated value of 0.1uA for the typical scenario at 25ºC as you can see in the image below. 0.1uA is so low that we don't need to power down the circuit at all as the battery will live ages with that current.
![]() |
0.1uA current at LPM4 mode |
Power supply measurement
The firmware also includes a measurement of the power supply after each signal sent. The ADC uses the 1.5V internal reference because the 2.5V one cannot be used below 2.8V. That means that our resistor divider used to measure the supply will get just the maximum 1023 10 bit count at 3V. The system detects the low voltage condition when the voltage falls below 2.6V, that means that the voltage at the divider is 1.3V so that the ADC gives about 890 counts.The firmware lights the red LED during 400ms after each transmited signal. When it detects a low battery voltage condition the LED is not turned on so that the user can see that the battery is nearly depleted.
Last words
That's all for now. The code currently ocupies about 2500 bytes with no optimization and with debug enabled. Disabling the debug and turning on the compiler optimizations its size falles to about 2000 bytes. That leaves a lot of room for future firmwares improvements as the MSP430G2553 features 16KB of flash memory.
In fact, the project could be developed in a smaller MCU but given the price differences in the MSP3250G budget line at low volumes there is little to gain in costs selecting another MCU.
Gist Update (30/11/2014)
Thanks to Gist, now the code, with syntax highlighting, is now included inside the blog entry.Code on Github (11/02/2018)
This new firmware and the old energia one are now on Github:https://github.com/R6500/Canon-IR-trigger