The general plan is to have an electromagnet whose magnetic field strength varies with the distance the magnetic levitating object, such that the distance is maintained constant. The distance from the levitating object is monitored with a Hall effect sensor. A second Hall effect sensor on the top of the electromagnet detects the magnetic field from the electromagnet and corrects the first sensor's reading to remove the influence of the electromagnet. The current through the coil is constantly adjusted to maintain a proper distance.
First, the electromagnet. I had a steel bar about 20 cm long by 1 cm diameter laying around. A soft iron core would have been better, but not free. I had several small spools of magnet wire, including 22, 26, and 30 AWG wire. The power supply I planned to use was a ATX power supply with 3.3V, 5V, and 12V taps. According to the Wikipedia page, magnet wire can typically handle 2.5-6 A/mm^2, depending on surroundings. Assuming the least conservative ampaciy, this corresponds to:
30 AWG: Current = 6 A/mm^2 * 0.0509 mm^2 = 0.3 A
200 feet at 103 ohm/1000 ft -> about 21 ohms resistance
V = I R = 0.3 * 21 = 6.3 V
26 AWG: Current = 6 A/mm^2 * 0.129 mm^2 = 0.8 A
75 feet at 40.81 ohm/1000 ft -> about 3.6 ohms resistance
V = 0.8 * 3.6 = 2.9 V
22 AWG: Current = 6 A/mm^2 * 0.326 mm^2 = 2.0 A
40 feet at 16.14 ohm/1000 ft -> about 0.65 ohms resistance
V = I R = 1.3 V
Ampere's law says that the strength of the magnetic field is proportional to the number of turns times the current through the wire. Since we want the maximum magnetic field we can get, if we multiply the lengths I had on hand by the max current in each wire, we get:
30 AWG : 0.3 A * 200 ft = 60
26 AWG : 0.8 A * 75 ft = 60
22 AWG : 2.0 A * 40 ft = 80
To get the strongest magnetic field, I should have chosen the 22 AWG wire. However, the length of wire I had has a max voltage of only 1.3 V, which would be rather difficult to control with a power supply with a minimum output of 3.3 volts. I'd need to keep the PWM down to a low level just to limit the current through the wire, leaving less room for control. The 30 AWG and 26 AWG wire should give the same magnetic field strength with voltages much closer to what the power supply puts out, so I arbitrarily chose the 30 AWG wire to use for the magnet.
I wrapped the steel bar with the 200 feet of the 30 AWG wire, which on a 1 cm diameter should have been a bit short of 2000 or so turns.
L = 200 ft
= 6096 cm
= # turns * circumference per turn
= # turns * Pi * diameter
# turns = 6096 cm/ Pi / 1 cm
= 1940 turns
Since there were several layers of wire on the bar, the actual number of turns is probably somewhat smaller, but should be good enough.
Loopalicious
Once I had the electromagnet coil wound, I made a wooden frame to mount the electromagnet in. I drilled a hole through a 1x2 and cut a slit through the hole and the wood. I then drilled a hole on the short side and put a machine screw through the hole, and attached a wing nut on the opposite end. This let me tighten down on the coil to hold it in place.
I placed the coil in the frame and attached two linear Hall effect sensors I'd bought from Digikey. These sensors output a 0-5V signal based on the strength of the magnetic field in their vicinity. The Hall sensor on the top of the electromagnet measures the strength of the magnetic field from the electromagnet. The Hall sensor measuring the distance to the floating magnet will also see the magnetic field from the electromagnet. Since we vary the strength of the electromagnet's field to keep the floating magnet steady, we need to compensate for this field to get a good estimate of the actual distance to the floating magnet. This could probably also be accomplished by running the electromagnet through its full current range and measuring the Hall reading at each point, then creating a calibration curve from these readings, but I chose to go the two-sensor route.
Tip - twist wires together with a drill to keep them organized. I first saw it here.
The circuit itself is pretty darn simple. It's just an n-Mosfet to control the coil, a flyback diode, two Hall sensors, and a potentiometer to adjust the distance between the floating magnet and the bottom of the electromagnet.
I'm proud of this soldering job. So fresh and so clean clean! Outkast would be proud, were he a worker in an electronics factory.
I used the Arduino PWM function to control the n-Mosfet to turn the coil on and off. The real heart of the project is the controller to take the Hall reading and output a PWM current to the coil. The controller has to work crazy fast, adjusting the current hundreds of times per second to keep the magnet stable. I ended up using a PID library to manage the calculation. Getting the PID parameters right took more time than the rest of the project combined. The final parameters have the coil power buzzing around like a pissed off yellowjacket, but at least it works!
I used the serial port to download Hall sensor readings and power output data to tweak the parameters. This chart is how the position of the magnet moves while floating with the final, working, PID parameters. I pulled the magnet out at the very end of this snapshot. The time axis on this chart is in milliseconds, so this entire span is only about a second and a half. Note how quickly the controller has to react to changes in the magnet position - it crosses the setpoint (of 150) about 40 times per second.
Another thing - the duty cycle of the coil while it's floating is only about 60%, meaning that, on average, the coil is only on about 60% of the time. This means that we can actually get away with a higher peak current through the coil than originally premised. Using the full 12V output from the power supply while the magnet is floating causes the coil to get a little warm, but definitely not hot enough to be concerning.
Here it is in action!
>#include <pid_v1.h> //PID control library
int Coil = 5; //PWM output pin that controls transistor for coil
double Setpoint, Input, Output; //PID variables
int PotPin = A2; // select the input pin for the potentiometer
int TimeConst = 1; //PID calculation frequency in milliseconds
int potValue = 0; // variable to store the value coming from the pot
//Specify the links and initial tuning parameters for PID controller
double Kp=5, Ki=4, Kd=.069;
PID myPID(&Input, &Output, &Setpoint, Kp, Ki, Kd, DIRECT);
int HALLA = 0; //Pin for reference hall effect sensor on top of coil
int HALLB = 1; //Pin for hall effect sensor on bottom of coil
int StopThreshold = 100; //Threshold below set point at which magnet will be turned off (if the floating magnet falls off or is removed, don't leave the coil just going full blast)
float CALSLOPE; //Slope of calibration curve for hall sensors
float CALINTERCEPT; //intercept of calibration curve for hall sensors
//variables for hall effect sensor calibraction
float CALSLOPEtemp = 0;
float CALINTERCEPTtemp = 0;
float XA;
float YA;
float XB;
float YB;
float CorrectedVal; //Compensated hall reading
int Power; //Coil PWM variable
void setup(){
//Reset PWM frequency to the coil pin to 62500/16 = 3906 Hz
setPwmFrequency(5, 16);
//Set PID sample time to defined value. Default is 200 ms, which is way too slow.
myPID.SetSampleTime(TimeConst);
Serial.begin(115200); //Need to transmit serial at 115200 bps to keep up
//Calibrate sensors by reading them when the coil is off, then reading them while coil is on. This allows you to compensate for the coil's magnetic field and just identify the magnetic field due to the floating magnet.
for (int i=0; i < 10; i++){
digitalWrite(Coil, LOW);
delay(100);
XA = analogRead(HALLA);
YA = analogRead(HALLB);
analogWrite(Coil, 255);
delay(100);
XB = analogRead(HALLA);
YB = analogRead(HALLB);
//Two point slope-intercept form
CALSLOPE = (YB-YA)/(XB-XA);
CALINTERCEPT = (YA-CALSLOPE * XA);
CALSLOPEtemp = CALSLOPEtemp + CALSLOPE;
CALINTERCEPTtemp = CALINTERCEPTtemp + CALINTERCEPT;
}
//Average slope and intercept
CALSLOPE = CALSLOPEtemp / 10;
CALINTERCEPT = CALINTERCEPTtemp/10;
Serial.println(CALSLOPE);
Serial.println(CALINTERCEPT);
digitalWrite(Coil, LOW);
//arbitrary starting setpoint. We'll reset it as soon as we enter the loop
Setpoint = 45;
//turn the PID on
myPID.SetMode(AUTOMATIC);
}
void loop(){
//Read potentiometer to get setpoint desired
potValue = analogRead(PotPin);
Setpoint = map(potValue, 0, 1023, 0, 255);
//Read hall effect sensors five times in a row, take average corrected reading
for (int i=0; i < 5; i++){
CorrectedVal =CorrectedVal + CALSLOPE * analogRead(HALLA) + CALINTERCEPT - analogRead(HALLB);
}
CorrectedVal = CorrectedVal / 5;
//Turn off coil if the magnet is nowhere nearby
if (CorrectedVal < Setpoint - StopThreshold) {
digitalWrite(Coil, LOW);
} else {
//Otherwise, do PID control to keep floating magnet at setpoint
Input =CorrectedVal;
myPID.Compute();
//Take PID output and apply to coil
Power = Output;
analogWrite(Coil, Power);
//Print out timestamp, coil power, corrected hall effect reading, and target set point for hall effect reading.
Serial.println(String(millis()) + ", "+String(int(Power)) + ", " + String(int(CorrectedVal))+ ", " + String(int(Setpoint)));
}
}
void setPwmFrequency(int pin, int divisor) {
byte mode;
if(pin == 5 || pin == 6 || pin == 9 || pin == 10) {
switch(divisor) {
case 1: mode = 0x01; break;
case 8: mode = 0x02; break;
case 64: mode = 0x03; break;
case 256: mode = 0x04; break;
case 1024: mode = 0x05; break;
default: return;
}
if(pin == 5 || pin == 6) {
TCCR0B = TCCR0B & 0b11111000 | mode;
} else {
TCCR1B = TCCR1B & 0b11111000 | mode;
}
} else if(pin == 3 || pin == 11) {
switch(divisor) {
case 1: mode = 0x01; break;
case 8: mode = 0x02; break;
case 32: mode = 0x03; break;
case 64: mode = 0x04; break;
case 128: mode = 0x05; break;
case 256: mode = 0x06; break;
case 1024: mode = 0x7; break;
default: return;
}
TCCR2B = TCCR2B & 0b11111000 | mode;
}
}