phase.c

Go to the documentation of this file.
00001 
00040 #include <drv/timer.h>
00041 
00042 
00043 #include <cfg/macros.h>
00044 #include <cpu/irq.h>
00045 #include <cpu/types.h>
00046 #include <cfg/compiler.h>
00047 
00048 
00049 #include <hw_phase.h>
00050 #include <drv/phase.h>
00051 
00052 #include <math.h>
00053 
00055 static Triac triacs[TRIAC_CNT];
00056 
00057 DB(bool phase_initialized;)
00058 
00066 DEFINE_ZEROCROSS_ISR()
00067 {
00068     ticks_t period, now;
00069     static ticks_t prev_time;
00070     TriacDev dev;
00071 
00072     now = timer_clock_unlocked();
00073     period = now - prev_time;
00074 
00075     for (dev = 0; dev < TRIAC_CNT; dev++)
00076     {
00077         /* Only turn off triac if duty is != 100% */
00078         if (triacs[dev].duty != TRIAC_MAX_DUTY)
00079             TRIAC_OFF(dev);
00080         /* Compute delay from duty */
00081         timer_setDelay(&triacs[dev].timer, DIV_ROUND(period * (TRIAC_MAX_DUTY - triacs[dev].duty), TRIAC_MAX_DUTY));
00082 
00083         /* This check avoids inserting the same timer twice
00084          * in case of an intempestive zerocross or spike */
00085         if (triacs[dev].running)
00086         {
00087             timer_abort(&triacs[dev].timer);
00088             //kprintf("[%lu]\n", timer_clock());
00089         }
00090 
00091         triacs[dev].running = true;
00092         timer_add(&triacs[dev].timer);
00093     }
00094     prev_time = now;
00095 }
00096 
00097 
00098 
00102 void phase_setDuty(TriacDev dev, triac_duty_t duty)
00103 {
00104     cpuflags_t flags;
00105     IRQ_SAVE_DISABLE(flags);
00106 
00107     phase_setDutyUnlock(dev,duty);
00108 
00109     IRQ_RESTORE(flags);
00110 }
00111 
00112 
00113 
00117 void phase_setDutyUnlock(TriacDev dev, triac_duty_t duty)
00118 {
00119     triacs[dev].duty = MIN(duty, (triac_duty_t)TRIAC_MAX_DUTY);
00120 }
00121 
00122 
00123 
00130 void phase_setPower(TriacDev dev, triac_power_t power)
00131 {
00132     bool greater_fifty = false;
00133     triac_duty_t duty;
00134 
00135     power = MIN(power, (triac_power_t)TRIAC_MAX_POWER);
00136 
00137     if (power > TRIAC_MAX_POWER / 2)
00138     {
00139         greater_fifty = true;
00140         power = TRIAC_MAX_POWER - power;
00141     }
00142 
00143     duty = TRIAC_POWER_K * sqrt(power);
00144 
00145     if (greater_fifty)
00146         duty = TRIAC_MAX_DUTY - duty;
00147     phase_setDuty(dev, duty);
00148 }
00149 
00150 
00151 
00157 static void phase_softint(void *_dev)
00158 {
00159     TriacDev dev = (TriacDev)_dev;
00160 
00161     /* Only turn on if duty is !=0 */
00162     if (triacs[dev].duty)
00163         TRIAC_ON(dev);
00164     triacs[dev].running = false;
00165 }
00166 
00167 
00168 
00172 void phase_init(void)
00173 {
00174     cpuflags_t flags;
00175     TriacDev dev;
00176 
00177     /* Init timers and ensure that all triac are off */
00178     for (dev = 0; dev < TRIAC_CNT; dev++)
00179     {
00180         triacs[dev].duty = 0;
00181         triacs[dev].running = false;
00182         SET_TRIAC_DDR(dev);
00183         TRIAC_OFF(dev);
00184         timer_set_event_softint(&triacs[dev].timer, (Hook)phase_softint, (void *)dev);
00185     }
00186     IRQ_SAVE_DISABLE(flags);
00187 
00188     /* Init zero cross interrupt */
00189     PHASE_HW_INIT;
00190     DB(phase_initialized = true;)
00191     IRQ_RESTORE(flags);
00192 }