This post explores how to use the power control features of the Battery Babysitter to make a product with the features we expect from portable embedded electronics.
In this modern age of cellphones, we've all come to expect a certain degree of elegance from our portable electronics' battery systems. But in DIY electronics it's common to find jumpers or switches that brutally disconnect the battery from its host. Or, no disconnect at all and puffy, discharged LiPos lying about haphazardly on the workbench. In my FLiR Pi Cam build, for instance, though voltage monitoring was built in, I left it to the user to isolate the batteries via a DPDT switch recessed in the body --- not very elegant at all!
This post explores how to use the power control features of the Battery Babysitter to make a product with the features we expect from portable embedded electronics.
This is an incident light meter I built inspired from another DIY project I found at kadookacameraworks.com. I built it partly because I wanted to try some menu stuff out on the screen, and partly because I could use it to help light scenes for photography.
Incident light meters need to be thrown in a camera bag and used on and off during a shoot, often from odd angles. It just wouldn't make sense to have a wire that needed to be plugged in, so this is a great platform to try out the Battery Babysitter in order to make a product that performs as people expect.
The "actual" functions of the device are really just obtained by connecting the TSL2561 sensor to the I2C bus and whacking on a TeensyView, followed by coding, coding and more coding. And maybe a little math mixed in there. At the end I'll leave you with a schematic and repository, but this isn't about how light is measured; it's about batteries and usability. So let's get to it!
It's a good idea to specify what you want in the beginning when designing something. Here's what I came up with for adding battery support to my project.
Before stuffing the project in the box, the entire project was developed on a breadboard. This allows the circuits in question to be tested and the construction to be well thought-out, avoiding sudden last-minute changes to the design after mechanical production has begun.
If we can describe how each goal is met, and how we can test each goal in the prototype, we can have confidence that the circuit will work as expected. So let's do that now.
Pouring through the Battery Babysitter's schematic, power flow goes first through the fuel gauge IC, through the charger IC, then to the user. Looking for off currents in the respective datasheets,
Ideally, I expect less than 5uA of drain during off time, which would take years to drain a 1000mAh battery. In reality, I found that the process of shutting down the fuel gauge was quite involved, and it's intended to operate automatically. Conceding that design point, the measured battery off current came to about 60uA. At 1000mAh, that's almost two years of off time before the battery goes dead. I'll expect to charge this every couple months.
To make the graphics of the battery (and the screen in general), I used a bitmap editor extensively. Ok, you got me; I used mspaint with the zoom all the way in and the grid enabled. Then, I converted the pixels into HEX, knowing that the MSb is on top. For instance, the plug starts with 0x28, 0x28, 0x7C... The graphics are encoded into functions like drawPlug
and batteryStyle1(x, y, %)
so they can be easily used.
To detect charge state, it's easy to look for a positive current flow into the battery. If the condition is met, the plug is drawn. To prevent constant drawing on the screen, though, I check the current and percent remaining by the Babysitter's SOC (state of charge) function and only update if there is a new value to display.
The idea of the SYSOFF pin is that it floats high (through a 5meg resistor) and needs to be pulled low in order to get the Babysitter to come online. By connecting two sink paths to the pin, essentially a NOR gate is created.
The first path is when the button is pressed, SYSOFF is pulled low through the diode. The second path occurs when the GPIO line is driven high, turning on the BJT to pull SYSOFF low.
The theory of operation is that when users want to use the system, they hold the power button until the processor boots. The first thing the processor does is configure its power control pin and pull SYSOFF low. Now the user can let go of the button, and the system remains on. When the system wants to shut itself down, it can release the power control line. The SYSOFF line goes high, and everyone goes to sleep.
The diode is included to allow the button to function normally on its input pin while the SYSOFF line is being held low by the processor.
This can be done by bailing from the menu state machine if an mstick count has expired. Not yet implemented. For now, power off occurs when the soft key is hit in the system menu.
To make the system 'catch' as soon as possible, the power pin is configured and asserted as the first thing within begin().
void setup()
{
//Assert power on pin
pinMode(PIN_POWER_ON, OUTPUT);
digitalWrite(PIN_POWER_ON, HIGH);
//...
Then, when I want to turn the whole thing off, I simply drive the pin low in response to a button operation. There's a bit more here to show the user what the system is doing, and to provide an out if the system fails to shut down.
if( encButton.serviceRisingEdge() )
{
oled.clear(PAGE);
oled.setFontType(1);
oled.setCursor(6, 6);
oled.print("Power down.");
oled.display();
delay(1000);
digitalWrite(PIN_POWER_ON, LOW);
delay(4000);
oled.clear(PAGE);
oled.setCursor(6, 6);
oled.print("Failed!!!");
oled.display();
delay(1000);
oled.setFontType(0);
oled.clear(PAGE);
oled.display();
nextState = pSystemInit;
}
As an example of graphics drawing, here's what the battery widget function does. First, it writes pixels to the screen memory with a bytewise draw command, and then it fills in the battery area based on the input percent.
void OLEDFunctions::batteryStyle1( uint8_t xIn, uint8_t yIn, float percent )
{
drawByte(xIn + 0,yIn + 0,0xFE);
drawByte(xIn + 1,yIn + 0,0x82);
drawByte(xIn + 2,yIn + 0,0x82);
drawByte(xIn + 3,yIn + 0,0x82);
drawByte(xIn + 4,yIn + 0,0x82);
drawByte(xIn + 5,yIn + 0,0x82);
drawByte(xIn + 6,yIn + 0,0x82);
drawByte(xIn + 7,yIn + 0,0x82);
drawByte(xIn + 8,yIn + 0,0x82);
drawByte(xIn + 9,yIn + 0,0xEE);
drawByte(xIn + 10,yIn + 0,0x38);
line(xIn + 2, yIn + 2, xIn + 2 + (percent * 6), yIn + 2);
line(xIn + 2, yIn + 3, xIn + 2 + (percent * 6), yIn + 3);
line(xIn + 2, yIn + 4, xIn + 2 + (percent * 6), yIn + 4);
};
All graphic functions in this project exist in a class that inherits the TeensyView object from the library. This separates the complicated drawing commands from the complicated state machine. The decoupling of these two things really helps to allow the firmware to be expanded --- for example, when a new menu screen needs to be created.
Take a look at this schematic to see how everything is wired together. If you're interested in actually making this project, or just seeing how it works, the code, schematic and 3D printer models can be found in a GitHub repository.
You can see that the Babysitter's output voltage drives the Teensy's bulk input (Vusb), which is regulated down to 3.3V for the Teensy and attached systems. Also, the USB communication lines have been extended to the convenient exposed PTH pads of the Babysitter, so the Teensy can be reprogrammed with the case closed.
The mechanical build is designed so the battery slides between a couple of snappable proto boards, and the whole assembly slides into the enclosure. No case screws!
Parts used in this build:
And again, the repository can be found here:
Please share your thoughts and ideas in the comments below.
Happy making!
---Marshall