Friday, September 24, 2021

0000 0000 0111 0100

Squeezing down the power of a PFS154 until it gasps

There are days when I wish that I had never heard of the magnificent reverse engineering project on EEVBlog for the Padauk range of microcontrollers. It is a wonderful piece of work put together by a talented group of individuals, but it is also a very deep rabbit hole from which it is difficult to escape.

There are other days when I claw back some sanity with a little win over this processor. It is often after trawling through datasheets, blogs and forums and piecing together a few snippets of code or advice that I have these tiny victories. 

I've always got my eyes on the prize of having solar "candles" driven in part by these "cheap" devices. I write "cheap" even though the cost of time and components for the programmers is surely ridiculous (and I cheerfully avoid any damning tallies).

Here are some of the requirements that have contributed to this goal:

Feature Blog Video
Designing a circuit that collects solar energy and can deliver that power reliably to a microcontroller


Blog Link

 


YT Link

 

Choosing an LED that is cheap, small, allows three channels and puts out a candle-like light Blog Link YT Link
Changing the energy requirements of a microcontroller for efficiency (lower frequency and voltage) Blog Link YT Link
Making a Programmer that can code up a Padauk chip Blog Link YT Link
Experiments in Random numbers to mimic a candle flame Blog Link YT Link 
Working with PWM on the PFS154 Blog Link YT Link 
8-bit AVR code upgraded to 16-bit for the PWM output of the PFS154 Blog Link  YT Link 
Can I make a PFS154 work? (not strictly candle based, but part of exploring this chip) Blog Link YT Link 

One of the joys of the ATTiny13 based candle is the very low energy requirements. I was able to run the chip with minimal energy requirements by lowering the clock frequency (128Mhz) and the voltage (3V). Now we are at a similar point with the Padauk version.

So I was chuffed to see Tim on his blog talking up the low energy blinker he made from the PFS154 - complete with code!

/* ---------------------------------------------------
	Ultra Low Power LED flasher
	LED is connected to PA4 and is high active.
	Jan 16th, 2021,  CPLDCPU - Initial version
  ---------------------------------------------------- */
  
#include <stdint.h>
#include <pdk/device.h>
#include <pdk/delay.h>

unsigned char _sdcc_external_startup(void)
{
	PDK_SET_FUSE(FUSE_IO_DRV_LOW);	// Set output driving strength to low
	
	CLKMD =  CLKMD_ILRC | CLKMD_ENABLE_ILRC | CLKMD_ENABLE_IHRC; 
	CLKMD =  CLKMD_ILRC | CLKMD_ENABLE_ILRC ; 

		// Note: it is important to turn off IHRC only after clock settings
		// have been updated. Otherwise the CPU stalls.
	return 0; // perform normal initialization
}

void main(void)
{

	// Toggling PA4 at ILRC (53kHz): 86 µA
	// Toggling PA4 at ILRC_dev16 (53kHz): 66.7 µA
	// Stopsys with 5V: 0.5 µA
	// The watchdog cannot wake up from stopexe!!

	PADIER = 0; // disable pins as wakeup source
	PBDIER = 0; // Also port B needs to be disabled even if it is not connected
		    // to the outside of the package. touching the package can introduce 
		    // glitches and wake up the device

	INTEN = 0;  // Make sure all interrupts are disabled
	INTRQ = 0;

	MISC = MISC_FAST_WAKEUP_ENABLE;  // Enable faster wakeup (45 clocks instead of 3000)
									 // This is important to not waste energy, as 40µA bias is already added during wakeup time

	// Configure timer two for output on PA3
	// --> Works, timer2 and 2 can be used for PWM while CPU is in STOPEXE mode
	// It appears that also timer 2 can wake up the CPU. This is not maskable?

	TM2C  = TM2C_CLK_ILRC | TM2C_MODE_PWM;  // Oscilator source for timer 2 is LRC (53 kHZ)
	TM2CT = 0;
	TM2S  = TM2S_PRESCALE_DIV16 | TM2S_SCALE_DIV8; // Divide clock by 16*7=112 -> 53 kHz / 122 = 414 Hz    
	TM2B  = 1;  // PWM threshold set to 1. The PWM event will trigger the wakeup. Wakeup occurs with 414 Hz / 256 = 1.66 Hz

	PA		= 1<<4;  // LED is on PA4, set all other output to zero.
	PAPH 	= 0;	         // Disable all pull up resistors
	PAC		= 0;     // Disable all outputs
				 // Note: There is no series resistor for the LED
				 // The LED current is limited to 2-4 mA by LOW IO driving setting
				 // See Setction 4.14 (p24) in PFS154 manual

	for (;;) {	
		PAC |=1<<4;  // Enable LED output (It's set to High)
		__nop();
		__nop();
		__nop();
		PAC &=~(1<<4); // Disable LED output after 4 cycles => 4/53 kHz = 75.5 µS
		__stopexe();
	}
}

I did make one small change in the end (not in the video) where I add a delay based on about 50 "nop" instructions so that the time "on" is greater (but also it uses more energy).

After wiring up the circuit and flashing the code...success! Low energy flashing LED from the PFS154.



Saturday, September 18, 2021

0000 0000 0111 0011

Rocking the "baby" (L298N II)

We've seen proof of concept for rocking an object using a modified UNO running an ATMega88 instead of an ATMega328, then controlling an L298N motor module which outputs to a dinky little hobbyist robot motor.

The big question is - can it be scaled up to a 12V automotive wiper motor?

The "specs" from my colleague on this project for the type of motion required are - a few seconds of rocking, a few minutes of no motion, control over speed and direction. 

The resulting code is as follows:

/*
   "Rocking the baby"
   Using a modified (ATMega88) Arduino to activate a 12V wiper
   motor via an L298N motor controller off Pins 7-9
   Modified Arduino: https://youtu.be/cfPpaX8V4S8

   The code will need the LowPower library available at:
   https://github.com/rocketscream/Low-Power
   (or you can write your own sleep routines)

   Also needed if you use ATMega88 are two things:
   1) MicroCore from https://github.com/MCUdude/MiniCore
   2) modify LowPower.h by adding ATMega88P to list

   This project can of course use a normal Arduino Uno or Nano etc

   Sketch uses 5498 bytes (67%) of program storage space. Maximum is 8192 bytes.
   Global variables use 81 bytes (7%) of dynamic memory,
   leaving 943 bytes for local variables. Maximum is 1024 bytes.

   A N Peck Mon 02 Aug 2021 19:00:12 AEST
   https://www.youtube.com/channel/UCTm3eCrN6wk-Ozt9ymyKIIg/videos
   https://onecircuit.blogspot.com/
*/

#include "LowPower.h"

// setup variables/constants

// Motor connections (pin numbers)
int enA = 9;
int in1 = 8;
int in2 = 7;
int LEDPin = 6;     // visual indicator of rocking
int BuzzPin = 5;    // audio indicator of rocking

int randomseed = 2901;      // changed in setup via analogRead(A5)

// change these to alter the rocking speed, duration and frequency
byte mspeed = 255;          // motorspeed (0-255)
int l_rocktime = 10;        // lower rocking time in s
int u_rocktime = 30;        // upper rocking time in s
int l_sleeptime = 8;        // lower sleeptime in min
int u_sleeptime = 12;       // lower sleeptime in min
String rotation = "clock";  // "clock" or "counter"

void rock(int rtime) {

  // warning - about to rock
  for (int count = 0; count < 5; count++) {
    tone(BuzzPin, 3200);
    digitalWrite(LEDPin, HIGH);
    delay(500);
    noTone(BuzzPin);
    digitalWrite(LEDPin, LOW);
    delay(1500);
  }

  if (rotation = "clock") {
    digitalWrite(in1, LOW);
    digitalWrite(in2, HIGH);
  }
  else {
    digitalWrite(in1, HIGH);
    digitalWrite(in2, LOW);
  }

  analogWrite(enA, mspeed);

  rtime = rtime / 2; // two second cycle below

  for (int ocount = 0; ocount < rtime; ocount++) {
    // two seconds of rotation, blinking and buzzing
    digitalWrite(LEDPin, HIGH);
    tone(BuzzPin, 3200);
    delay(100);
    noTone(BuzzPin);
    delay(100);
    tone(BuzzPin, 2500);
    delay(200);
    noTone(BuzzPin);
    delay(600);
    digitalWrite(LEDPin, LOW);
    delay(1000);
  }

  // turn off motor
  digitalWrite(in1, LOW);
  digitalWrite(in2, LOW);
}

void gosleep(int stime) {
  stime = stime * 60; // minutes to seconds
  stime = stime / 8; // seconds to multiples of 8

  for (int count = 0; count < stime; count++) {
    LowPower.powerDown(SLEEP_8S, ADC_OFF, BOD_OFF);
  }
}

void setup() {
  // randomseed changed
  randomseed = analogRead(A5);

  // Set all the control pins to outputs
  pinMode(enA, OUTPUT);
  pinMode(in1, OUTPUT);
  pinMode(in2, OUTPUT);
  pinMode(LEDPin, OUTPUT);
  pinMode(BuzzPin, OUTPUT);

  // Turn off motors - Initial state
  digitalWrite(in1, LOW);
  digitalWrite(in2, LOW);
}

void loop() {

  int rocktime = random(l_rocktime, u_rocktime);  // time rocking (s)
  rock(rocktime);

  int wtime = random(l_sleeptime, u_sleeptime);   // time asleep (min)
  gosleep(wtime);

}

I added an LED and a BUZZER to alert all around that the rocking is about to start (Work, Health and Safety!) - and apart from that all that was needed was the identification of wires on the motor and we were good to go.


Saturday, September 11, 2021

0000 0000 0111 0010

120W bluetooth sound system

I have built an amp from an LM386 before, documented on this blog and associated YouTube video.

It faithfully churns out around 1W of sound when you use an LM386-4, and is a pretty handy (and super cheap) option for amplification.

However, there may come a time when you want to crank out some phat beats (I kid you not...) or indeed kick back and absorb some Octaviated Hairpins, and then you might need more grunt out of your amp than the LM386 can provide.

Furthermore, you also might want to swap out a topographically restrictive audio lead for a BlueTooth device which will provide a bit more portability and easy control over your tunes.

Enter (from my recent mailbag) a bluetooth module and a promising amp module based on the TPA3116 chip.

Firstly, I pulled apart a reclaimed headphone cable for the 3.5mm audio jack, and found that even though there were only three zones on the jack, there were four wires!

With a little experimentation I determined that the wires were LEFT, RIGHT and two GND wires.

Hooking up 12V (it can take up to 24V) I was able to get some good noise out of the amp, although it took a little to sort out the left/right wiring.

Once up and running, it was fairly trivial to slide in the BlueTooth module and get some music blaring out from my phone to the recycled speakers.


This is an Amp/BlueTooth combo that I would definitely recommend - it was very simple to set up, and also a very very cheap option considering the resultant sound. The TPA3116 is now one of my favourites!

I am going to 3D print a box to put the modules in, and power it with a 12V adapter (maybe 2A), and then I'm good to rock in the shop.



Friday, September 3, 2021

0000 0000 0111 0001

L298N motor control module

A colleague of mine challenged me to come up with a "rocking device" capable of moving a fairly large cylindrical object (not a baby...) backwards and forwards every few minutes.

The coding part is fairly simple:

#include <LowPower.h>

// Motor A connections
int enA = 9;
int in1 = 8;
int in2 = 7;

byte mspeed = 255;
int mtime = 1600;
int wtime = 5000;

void setup() {
  // Set all the motor control pins to outputs
  pinMode(enA, OUTPUT);
  pinMode(in1, OUTPUT);
  pinMode(in2, OUTPUT);

  // Turn off motors - Initial state
  digitalWrite(in1, LOW);
  digitalWrite(in2, LOW);

  LowPower.powerDown(SLEEP_8S, ADC_OFF, BOD_OFF);
}

void loop() {

  digitalWrite(in1, LOW);
  digitalWrite(in2, HIGH);
  analogWrite(enA, mspeed);
  delay(mtime);

  digitalWrite(in1, LOW);
  digitalWrite(in2, LOW);

  delay(wtime);

  digitalWrite(in1, HIGH);
  digitalWrite(in2, LOW);

  analogWrite(enA, mspeed);
  delay(mtime);

  digitalWrite(in1, LOW);
  digitalWrite(in2, LOW);

  delay(wtime);

}


I decided to repurpose a "broken" UNO by replacing the fried ATMega328 with a new ATMega88 as per this blog post and video.

When it comes to translating mA control pulses from such a micro-controller it becomes necessary to design either some amplification/protection circuit, or simply use a purpose built module based around the L298N IC. From the datasheet:

The L298 is an integrated monolithic circuit. It is a high voltage, high current dual full-bridge driver designed to accept standard TTL logic levels and drive inductive loads such as relays, solenoids, DC and stepping motors.

I happened to have a module based around this IC which looked perfect for this application (thanks buckets).

My first attempt using a 12V motor was a bust as it was rotating too fast (not geared). Even when I used a geared motor, there were issues with overcoming friction to get the motor in motion.

I solved this by "blipping" the throttle a little bit before the motion starts, in order to get the rotation happening. This in code is a simple staggered start with the motor given almost the full juice for 20ms and then two thirds of that again for 10ms before the final controlled amount (e.g. 25/255 steps).

The final code didn't need this because the gearing was more forgiving.

Now that the prototype is working fine, I'm just waiting on a full sized motor (probably a salvaged windscreen wiper) to complete the challenge. Nice project!