module System.Hardware.Arduino.Firmata.Basics where
import Control.Concurrent (newEmptyMVar, readMVar)
import Control.Monad (when)
import Control.Monad.Trans (liftIO)
import Data.Word (Word8)
import System.Hardware.Arduino.Data
import System.Hardware.Arduino.Comm
import qualified System.Hardware.Arduino.Utils as U
queryFirmware :: Arduino (Word8, Word8, String)
queryFirmware = do
send QueryFirmware
r <- recv
case r of
Firmware v1 v2 m -> return (v1, v2, m)
_ -> error $ "queryFirmware: Got unexpected response for query firmware call: " ++ show r
delay :: Int -> Arduino ()
delay = liftIO . U.delay
setPinMode :: Pin -> PinMode -> Arduino ()
setPinMode p m = do
extras <- registerPinMode p m
send $ SetPinMode p m
mapM_ send extras
digitalWrite :: Pin -> Bool -> Arduino ()
digitalWrite p v = do
pd <- getPinData p
when (pinMode pd /= OUTPUT) $ U.die ("Invalid digitalWrite call on pin " ++ show p)
[ "The current mode for this pin is: " ++ show (pinMode pd)
, "For digitalWrite, it must be set to: " ++ show OUTPUT
, "via a proper call to setPinMode"
]
(lsb, msb) <- computePortData p v
send $ DigitalPortWrite (pinPort p) lsb msb
digitalRead :: Pin -> Arduino Bool
digitalRead p = do
pd <- getPinData p
when (pinMode pd /= INPUT) $ U.die ("Invalid digitalRead call on pin " ++ show p)
[ "The current mode for this pin is: " ++ show (pinMode pd)
, "For digitalWrite, it must be set to: " ++ show INPUT
, "via a proper call to setPinMode"
]
return $ case pinValue pd of
Just (Left v) -> v
_ -> False
waitFor :: Pin -> Arduino Bool
waitFor p = do
curVal <- digitalRead p
let wait = do sleepTillDigitalMessage
newVal <- digitalRead p
if newVal == curVal
then wait
else return newVal
wait
sleepTillDigitalMessage :: Arduino ()
sleepTillDigitalMessage = do
semaphore <- liftIO newEmptyMVar
digitalWakeUp semaphore
liftIO $ readMVar semaphore