module Data.ASN1.BitArray
	( BitArray(..)
	, BitArrayOutOfBound(..)
	, bitArrayLength
	, bitArrayGetBit
	, bitArraySetBitValue
	, bitArraySetBit
	, bitArrayClearBit
	, bitArrayGetData
	, toBitArray
	) where
import Data.Bits
import Data.Word
import Data.Maybe
import qualified Data.ByteString.Lazy as L
import Data.Typeable
import Control.Exception (Exception, throw)
data BitArrayOutOfBound = BitArrayOutOfBound Word64
	deriving (Show,Eq,Typeable)
instance Exception BitArrayOutOfBound
data BitArray = BitArray Word64 L.ByteString
	deriving (Show,Eq)
bitArrayLength :: BitArray -> Word64
bitArrayLength (BitArray l _) = l
bitArrayOutOfBound :: Word64 -> a
bitArrayOutOfBound n = throw $ BitArrayOutOfBound n
bitArrayGetBit :: BitArray -> Word64 -> Bool
bitArrayGetBit (BitArray l d) n
	| n >= l    = bitArrayOutOfBound n
	| otherwise = flip testBit (7fromIntegral bitn) $ L.index d (fromIntegral offset)
		where (offset, bitn) = n `divMod` 8
bitArraySetBitValue :: BitArray -> Word64 -> Bool -> BitArray
bitArraySetBitValue (BitArray l d) n v
	| n >= l    = bitArrayOutOfBound n
	| otherwise =
		let (before,after) = L.splitAt (fromIntegral offset) d in
		
		let (w,remaining) = fromJust $ L.uncons after in
		BitArray l (before `L.append` (setter w (fromIntegral bitn) `L.cons` remaining))
		where
			(offset, bitn) = n `divMod` 8
			setter = if v then setBit else clearBit
bitArraySetBit :: BitArray -> Word64 -> BitArray
bitArraySetBit bitarray n = bitArraySetBitValue bitarray n True
bitArrayClearBit :: BitArray -> Word64 -> BitArray
bitArrayClearBit bitarray n = bitArraySetBitValue bitarray n False
bitArrayGetData :: BitArray -> L.ByteString
bitArrayGetData (BitArray _ d) = d
toBitArray :: L.ByteString -> Int -> BitArray
toBitArray l toSkip =
	BitArray (fromIntegral (L.length l * 8  fromIntegral toSkip)) l