{-# LANGUAGE OverloadedStrings, BangPatterns, ScopedTypeVariables #-}
module Crypto.PHKDF
( HmacKey()
, hmacKey
, PhkdfCtx()
, phkdfCtx
, phkdfCtx_init
, phkdfCtx_initHashed
, phkdfCtx_initPrefixed
, phkdfCtx_initLike
, phkdfCtx_hmacKeyPlain
, phkdfCtx_hmacKeyHashed
, phkdfCtx_hmacKeyPrefixed
, phkdfCtx_hmacKey
, phkdfCtx_hmacKeyLike
, phkdfCtx_toResetHmacCtx
, phkdfCtx_reset
, phkdfCtx_feedArg
, phkdfCtx_feedArgs
, phkdfCtx_feedArgsBy
, phkdfCtx_feedArgConcat
, phkdfCtx_finalize
, phkdfCtx_finalizeHmac
, phkdfCtx_toHmacCtx
, phkdfCtx_toHmacKeyPrefixed
, phkdfCtx_toStream
, phkdfCtx_toGen
, phkdfCtx_byteCount
, phkdfCtx_endPaddingLength
, phkdfCtx_blockPaddingLength
, PhkdfGen()
, phkdfGen
, phkdfGen_init
, phkdfGen_initHashed
, phkdfGen_initPrefixed
, phkdfGen_initLike
, phkdfGen_hmacKeyPlain
, phkdfGen_hmacKeyHashed
, phkdfGen_hmacKeyPrefixed
, phkdfGen_hmacKey
, phkdfGen_hmacKeyLike
, phkdfGen_head
, phkdfGen_read
, phkdfGen_peek
, phkdfGen_toStream
) where
import Control.Arrow((>>>))
import Data.Bits((.&.), complement)
import Data.ByteString (ByteString)
import qualified Data.ByteString as B
import Data.Function((&))
import Data.Foldable(Foldable, foldl')
import Data.Word
import Data.Stream (Stream(..))
import qualified Data.Stream as Stream
import Network.ByteOrder (bytestring32)
import Crypto.Sha256 as Sha256
import Crypto.PHKDF.HMAC
import Crypto.PHKDF.HMAC.Subtle
import Crypto.PHKDF.Subtle
import Crypto.Encoding.PHKDF
import Crypto.Encoding.SHA3.TupleHash
import Control.Exception(assert)
phkdfCtx :: ByteString -> PhkdfCtx
phkdfCtx :: ByteString -> PhkdfCtx
phkdfCtx = HmacKey -> PhkdfCtx
phkdfCtx_init (HmacKey -> PhkdfCtx)
-> (ByteString -> HmacKey) -> ByteString -> PhkdfCtx
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> HmacKey
hmacKey
phkdfCtx_init :: HmacKey -> PhkdfCtx
phkdfCtx_init :: HmacKey -> PhkdfCtx
phkdfCtx_init = HmacKeyLike -> PhkdfCtx
phkdfCtx_initLike (HmacKeyLike -> PhkdfCtx)
-> (HmacKey -> HmacKeyLike) -> HmacKey -> PhkdfCtx
forall b c a. (b -> c) -> (a -> b) -> a -> c
. HmacKey -> HmacKeyLike
hmacKeyLike_init
phkdfCtx_initLike :: HmacKeyLike -> PhkdfCtx
phkdfCtx_initLike :: HmacKeyLike -> PhkdfCtx
phkdfCtx_initLike HmacKeyLike
key =
PhkdfCtx {
phkdfCtx_byteCount :: Word64
phkdfCtx_byteCount = HmacKeyLike -> Word64
hmacKeyLike_byteCount HmacKeyLike
key,
phkdfCtx_state :: Sha256Ctx
phkdfCtx_state = HmacKeyLike -> Sha256Ctx
hmacKeyLike_ipadCtx HmacKeyLike
key,
phkdfCtx_hmacKeyLike :: HmacKeyLike
phkdfCtx_hmacKeyLike = HmacKeyLike
key
}
phkdfCtx_initHashed :: HmacKeyHashed -> PhkdfCtx
phkdfCtx_initHashed :: HmacKeyHashed -> PhkdfCtx
phkdfCtx_initHashed = HmacKey -> PhkdfCtx
phkdfCtx_init (HmacKey -> PhkdfCtx)
-> (HmacKeyHashed -> HmacKey) -> HmacKeyHashed -> PhkdfCtx
forall b c a. (b -> c) -> (a -> b) -> a -> c
. HmacKeyHashed -> HmacKey
hmacKeyHashed_toKey
phkdfCtx_initPrefixed :: ByteString -> HmacKeyPrefixed -> PhkdfCtx
phkdfCtx_initPrefixed :: ByteString -> HmacKeyPrefixed -> PhkdfCtx
phkdfCtx_initPrefixed ByteString
str HmacKeyPrefixed
key = PhkdfCtx
{ phkdfCtx_byteCount :: Word64
phkdfCtx_byteCount = HmacKeyPrefixed -> Word64
hmacKeyPrefixed_byteCount HmacKeyPrefixed
key
Word64 -> Word64 -> Word64
forall a. Num a => a -> a -> a
+ Int -> Word64
forall a b. (Integral a, Num b) => a -> b
fromIntegral (ByteString -> Int
B.length ByteString
str)
, phkdfCtx_state :: Sha256Ctx
phkdfCtx_state = Sha256Ctx -> ByteString -> Sha256Ctx
sha256_update (HmacKeyPrefixed -> Sha256Ctx
hmacKeyPrefixed_ipadCtx HmacKeyPrefixed
key) ByteString
str
, phkdfCtx_hmacKeyLike :: HmacKeyLike
phkdfCtx_hmacKeyLike = HmacKeyPrefixed -> HmacKeyLike
hmacKeyLike_initPrefixed HmacKeyPrefixed
key
}
phkdfCtx_hmacKeyPlain :: PhkdfCtx -> Maybe HmacKeyPlain
phkdfCtx_hmacKeyPlain :: PhkdfCtx -> Maybe ByteString
phkdfCtx_hmacKeyPlain = HmacKeyLike -> Maybe ByteString
hmacKeyLike_toPlain (HmacKeyLike -> Maybe ByteString)
-> (PhkdfCtx -> HmacKeyLike) -> PhkdfCtx -> Maybe ByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfCtx -> HmacKeyLike
phkdfCtx_hmacKeyLike
phkdfCtx_hmacKeyHashed :: PhkdfCtx -> Maybe HmacKeyHashed
phkdfCtx_hmacKeyHashed :: PhkdfCtx -> Maybe HmacKeyHashed
phkdfCtx_hmacKeyHashed = HmacKeyLike -> Maybe HmacKeyHashed
hmacKeyLike_toHashed (HmacKeyLike -> Maybe HmacKeyHashed)
-> (PhkdfCtx -> HmacKeyLike) -> PhkdfCtx -> Maybe HmacKeyHashed
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfCtx -> HmacKeyLike
phkdfCtx_hmacKeyLike
phkdfCtx_hmacKeyPrefixed :: PhkdfCtx -> HmacKeyPrefixed
phkdfCtx_hmacKeyPrefixed :: PhkdfCtx -> HmacKeyPrefixed
phkdfCtx_hmacKeyPrefixed = HmacKeyLike -> HmacKeyPrefixed
hmacKeyLike_toPrefixed (HmacKeyLike -> HmacKeyPrefixed)
-> (PhkdfCtx -> HmacKeyLike) -> PhkdfCtx -> HmacKeyPrefixed
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfCtx -> HmacKeyLike
phkdfCtx_hmacKeyLike
phkdfCtx_hmacKey :: PhkdfCtx -> Maybe HmacKey
phkdfCtx_hmacKey :: PhkdfCtx -> Maybe HmacKey
phkdfCtx_hmacKey = HmacKeyLike -> Maybe HmacKey
hmacKeyLike_toKey (HmacKeyLike -> Maybe HmacKey)
-> (PhkdfCtx -> HmacKeyLike) -> PhkdfCtx -> Maybe HmacKey
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfCtx -> HmacKeyLike
phkdfCtx_hmacKeyLike
phkdfCtx_reset :: PhkdfCtx -> PhkdfCtx
phkdfCtx_reset :: PhkdfCtx -> PhkdfCtx
phkdfCtx_reset = HmacKeyLike -> PhkdfCtx
phkdfCtx_initLike (HmacKeyLike -> PhkdfCtx)
-> (PhkdfCtx -> HmacKeyLike) -> PhkdfCtx -> PhkdfCtx
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfCtx -> HmacKeyLike
phkdfCtx_hmacKeyLike
phkdfCtx_toResetHmacCtx :: PhkdfCtx -> HmacCtx
phkdfCtx_toResetHmacCtx :: PhkdfCtx -> HmacCtx
phkdfCtx_toResetHmacCtx = HmacKeyLike -> HmacCtx
hmacKeyLike_run (HmacKeyLike -> HmacCtx)
-> (PhkdfCtx -> HmacKeyLike) -> PhkdfCtx -> HmacCtx
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfCtx -> HmacKeyLike
phkdfCtx_hmacKeyLike
phkdfCtx_feedArg :: ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArg :: ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArg ByteString
str = [ByteString] -> PhkdfCtx -> PhkdfCtx
forall (f :: * -> *).
Foldable f =>
f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_unsafeFeed [ByteString
len, ByteString
str]
where
len :: ByteString
len = Int -> ByteString
forall b. (Integral b, FiniteBits b) => b -> ByteString
leftEncodeFromBytes (ByteString -> Int
B.length ByteString
str)
phkdfCtx_feedArgs :: Foldable f => f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArgs :: forall (f :: * -> *).
Foldable f =>
f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArgs f ByteString
params PhkdfCtx
ctx = (PhkdfCtx -> ByteString -> PhkdfCtx)
-> PhkdfCtx -> f ByteString -> PhkdfCtx
forall b a. (b -> a -> b) -> b -> f a -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' ((ByteString -> PhkdfCtx -> PhkdfCtx)
-> PhkdfCtx -> ByteString -> PhkdfCtx
forall a b c. (a -> b -> c) -> b -> a -> c
flip ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArg) PhkdfCtx
ctx f ByteString
params
phkdfCtx_feedArgsBy :: Foldable f => (a -> ByteString) -> f a -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArgsBy :: forall (f :: * -> *) a.
Foldable f =>
(a -> ByteString) -> f a -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArgsBy a -> ByteString
f f a
params PhkdfCtx
ctx0 = (PhkdfCtx -> a -> PhkdfCtx) -> PhkdfCtx -> f a -> PhkdfCtx
forall b a. (b -> a -> b) -> b -> f a -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' PhkdfCtx -> a -> PhkdfCtx
delta PhkdfCtx
ctx0 f a
params
where delta :: PhkdfCtx -> a -> PhkdfCtx
delta PhkdfCtx
ctx a
a = ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArg (a -> ByteString
f a
a) PhkdfCtx
ctx
phkdfCtx_feedArgConcat :: Foldable f => f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArgConcat :: forall (f :: * -> *).
Foldable f =>
f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_feedArgConcat f ByteString
strs =
[ByteString] -> PhkdfCtx -> PhkdfCtx
forall (f :: * -> *).
Foldable f =>
f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_unsafeFeed [ByteString
len] (PhkdfCtx -> PhkdfCtx)
-> (PhkdfCtx -> PhkdfCtx) -> PhkdfCtx -> PhkdfCtx
forall {k} (cat :: k -> k -> *) (a :: k) (b :: k) (c :: k).
Category cat =>
cat a b -> cat b c -> cat a c
>>>
f ByteString -> PhkdfCtx -> PhkdfCtx
forall (f :: * -> *).
Foldable f =>
f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_unsafeFeed f ByteString
strs
where
len :: ByteString
len = Int -> ByteString
forall b. (Integral b, FiniteBits b) => b -> ByteString
leftEncodeFromBytes ((Int -> ByteString -> Int) -> Int -> f ByteString -> Int
forall b a. (b -> a -> b) -> b -> f a -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' Int -> ByteString -> Int
delta Int
0 f ByteString
strs)
delta :: Int -> ByteString -> Int
delta Int
tot ByteString
str = Int
tot Int -> Int -> Int
forall a. Num a => a -> a -> a
+ ByteString -> Int
B.length ByteString
str
phkdfCtx_finalize
:: (Int -> ByteString)
-> Word32
-> ByteString
-> PhkdfCtx
-> ByteString
phkdfCtx_finalize :: (Int -> ByteString)
-> Word32 -> ByteString -> PhkdfCtx -> ByteString
phkdfCtx_finalize Int -> ByteString
genFillerPad Word32
counter ByteString
tag PhkdfCtx
ctx =
(Int -> ByteString) -> Word32 -> ByteString -> PhkdfCtx -> PhkdfGen
phkdfCtx_toGen Int -> ByteString
genFillerPad Word32
counter ByteString
tag PhkdfCtx
ctx PhkdfGen -> (PhkdfGen -> ByteString) -> ByteString
forall a b. a -> (a -> b) -> b
&
PhkdfGen -> ByteString
phkdfGen_head
phkdfCtx_toHmacCtx :: PhkdfCtx -> HmacCtx
phkdfCtx_toHmacCtx :: PhkdfCtx -> HmacCtx
phkdfCtx_toHmacCtx PhkdfCtx
ctx =
(PhkdfCtx -> HmacCtx
phkdfCtx_toResetHmacCtx PhkdfCtx
ctx) {
hmacCtx_ipadCtx = phkdfCtx_state ctx
}
phkdfCtx_toHmacKeyPrefixed
:: (Int -> ByteString)
-> PhkdfCtx
-> HmacKeyPrefixed
phkdfCtx_toHmacKeyPrefixed :: (Int -> ByteString) -> PhkdfCtx -> HmacKeyPrefixed
phkdfCtx_toHmacKeyPrefixed Int -> ByteString
genFillerPad PhkdfCtx
ctx =
HmacKeyPrefixed
{ hmacKeyPrefixed_ipadCtx :: Sha256Ctx
hmacKeyPrefixed_ipadCtx = Sha256Ctx
ipadCtx'
, hmacKeyPrefixed_opad :: Sha256State
hmacKeyPrefixed_opad = HmacKeyLike -> Sha256State
hmacKeyLike_opad (PhkdfCtx -> HmacKeyLike
phkdfCtx_hmacKeyLike PhkdfCtx
ctx)
}
where
blockPadLen :: Int
blockPadLen = PhkdfCtx -> Int
phkdfCtx_blockPaddingLength PhkdfCtx
ctx
blockPadding :: ByteString
blockPadding = Int -> ByteString
genFillerPad Int
blockPadLen
ctx' :: PhkdfCtx
ctx' = [ByteString] -> PhkdfCtx -> PhkdfCtx
forall (f :: * -> *).
Foldable f =>
f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_unsafeFeed [ByteString
"\x00",ByteString
blockPadding] PhkdfCtx
ctx
paddingIsValid :: Bool
paddingIsValid = PhkdfCtx -> Word64
phkdfCtx_byteCount PhkdfCtx
ctx' Word64 -> Word64 -> Word64
forall a. Integral a => a -> a -> a
`mod` Word64
64 Word64 -> Word64 -> Bool
forall a. Eq a => a -> a -> Bool
== Word64
0
Bool -> Bool -> Bool
&& ByteString -> Int
B.length ByteString
blockPadding Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
blockPadLen
ipadCtx' :: Sha256Ctx
ipadCtx' = Bool -> Sha256Ctx -> Sha256Ctx
forall a. HasCallStack => Bool -> a -> a
assert Bool
paddingIsValid (Sha256Ctx -> Sha256Ctx) -> Sha256Ctx -> Sha256Ctx
forall a b. (a -> b) -> a -> b
$ PhkdfCtx -> Sha256Ctx
phkdfCtx_state PhkdfCtx
ctx'
phkdfCtx_finalizeHmac :: PhkdfCtx -> ByteString
phkdfCtx_finalizeHmac :: PhkdfCtx -> ByteString
phkdfCtx_finalizeHmac = HmacCtx -> ByteString
hmacCtx_finalize_toByteString (HmacCtx -> ByteString)
-> (PhkdfCtx -> HmacCtx) -> PhkdfCtx -> ByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfCtx -> HmacCtx
phkdfCtx_toHmacCtx
phkdfCtx_toStream :: (Int -> ByteString) -> Word32 -> ByteString -> PhkdfCtx -> Stream ByteString
phkdfCtx_toStream :: (Int -> ByteString)
-> Word32 -> ByteString -> PhkdfCtx -> Stream ByteString
phkdfCtx_toStream Int -> ByteString
genFillerPad Word32
counter0 ByteString
tag PhkdfCtx
ctx =
(Int -> ByteString) -> Word32 -> ByteString -> PhkdfCtx -> PhkdfGen
phkdfCtx_toGen Int -> ByteString
genFillerPad Word32
counter0 ByteString
tag PhkdfCtx
ctx PhkdfGen -> (PhkdfGen -> Stream ByteString) -> Stream ByteString
forall a b. a -> (a -> b) -> b
&
PhkdfGen -> Stream ByteString
phkdfGen_toStream
phkdfCtx_endPaddingLength :: PhkdfCtx -> Int
phkdfCtx_endPaddingLength :: PhkdfCtx -> Int
phkdfCtx_endPaddingLength PhkdfCtx
ctx =
Word64 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral ((Word64
31 Word64 -> Word64 -> Word64
forall a. Num a => a -> a -> a
- PhkdfCtx -> Word64
phkdfCtx_byteCount PhkdfCtx
ctx) Word64 -> Word64 -> Word64
forall a. Bits a => a -> a -> a
.&. Word64
63)
phkdfCtx_blockPaddingLength :: PhkdfCtx -> Int
phkdfCtx_blockPaddingLength :: PhkdfCtx -> Int
phkdfCtx_blockPaddingLength PhkdfCtx
ctx =
Word64 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral ((Word64
63 Word64 -> Word64 -> Word64
forall a. Num a => a -> a -> a
- PhkdfCtx -> Word64
phkdfCtx_byteCount PhkdfCtx
ctx) Word64 -> Word64 -> Word64
forall a. Bits a => a -> a -> a
.&. Word64
63)
phkdfCtx_toGen
:: (Int -> ByteString)
-> Word32
-> ByteString
-> PhkdfCtx
-> PhkdfGen
phkdfCtx_toGen :: (Int -> ByteString) -> Word32 -> ByteString -> PhkdfCtx -> PhkdfGen
phkdfCtx_toGen Int -> ByteString
genFillerPad Word32
counter0 ByteString
tag PhkdfCtx
ctx =
PhkdfGen
{ phkdfGen_hmacKeyLike :: HmacKeyLike
phkdfGen_hmacKeyLike = PhkdfCtx -> HmacKeyLike
phkdfCtx_hmacKeyLike PhkdfCtx
ctx
, phkdfGen_extTag :: ByteString
phkdfGen_extTag = ByteString -> ByteString
extendTag ByteString
tag
, phkdfGen_counter :: Word32
phkdfGen_counter = Word32
counter0
, phkdfGen_state :: ByteString
phkdfGen_state = ByteString
""
, phkdfGen_initCtx :: Maybe Sha256Ctx
phkdfGen_initCtx = Sha256Ctx -> Maybe Sha256Ctx
forall a. a -> Maybe a
Just Sha256Ctx
context0
}
where
endPadLen :: Int
endPadLen = PhkdfCtx -> Int
phkdfCtx_endPaddingLength PhkdfCtx
ctx
endPadding :: ByteString
endPadding = Int -> ByteString
genFillerPad Int
endPadLen
ctx' :: PhkdfCtx
ctx' = [ByteString] -> PhkdfCtx -> PhkdfCtx
forall (f :: * -> *).
Foldable f =>
f ByteString -> PhkdfCtx -> PhkdfCtx
phkdfCtx_unsafeFeed [ByteString
"\x00",ByteString
endPadding] PhkdfCtx
ctx
endPaddingIsValid :: Bool
endPaddingIsValid = PhkdfCtx -> Word64
phkdfCtx_byteCount PhkdfCtx
ctx' Word64 -> Word64 -> Word64
forall a. Integral a => a -> a -> a
`mod` Word64
64 Word64 -> Word64 -> Bool
forall a. Eq a => a -> a -> Bool
== Word64
32
Bool -> Bool -> Bool
&& ByteString -> Int
B.length ByteString
endPadding Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
endPadLen
context0 :: Sha256Ctx
context0 = Bool -> Sha256Ctx -> Sha256Ctx
forall a. HasCallStack => Bool -> a -> a
assert Bool
endPaddingIsValid (Sha256Ctx -> Sha256Ctx) -> Sha256Ctx -> Sha256Ctx
forall a b. (a -> b) -> a -> b
$ PhkdfCtx -> Sha256Ctx
phkdfCtx_state PhkdfCtx
ctx'
phkdfGen :: ByteString -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen :: ByteString -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen = HmacKey -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_init (HmacKey -> ByteString -> Word32 -> ByteString -> PhkdfGen)
-> (ByteString -> HmacKey)
-> ByteString
-> ByteString
-> Word32
-> ByteString
-> PhkdfGen
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> HmacKey
hmacKey
phkdfGen_init :: HmacKey -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_init :: HmacKey -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_init = HmacKeyLike -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initLike (HmacKeyLike -> ByteString -> Word32 -> ByteString -> PhkdfGen)
-> (HmacKey -> HmacKeyLike)
-> HmacKey
-> ByteString
-> Word32
-> ByteString
-> PhkdfGen
forall b c a. (b -> c) -> (a -> b) -> a -> c
. HmacKey -> HmacKeyLike
hmacKeyLike_init
phkdfGen_initLike :: HmacKeyLike -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initLike :: HmacKeyLike -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initLike HmacKeyLike
key ByteString
initBytes = Word32 -> ByteString -> PhkdfGen
initGen
where
n :: Int
n = ByteString -> Int
B.length ByteString
initBytes Int -> Int -> Int
forall a. Bits a => a -> a -> a
.&. Int -> Int
forall a. Bits a => a -> a
complement Int
63
(ByteString
blocks, ByteString
state0) = Int -> ByteString -> (ByteString, ByteString)
B.splitAt Int
n ByteString
initBytes
ipad0 :: Sha256Ctx
ipad0 = Sha256Ctx -> ByteString -> Sha256Ctx
sha256_update (HmacKeyLike -> Sha256Ctx
hmacKeyLike_ipadCtx HmacKeyLike
key) ByteString
blocks
initGen :: Word32 -> ByteString -> PhkdfGen
initGen Word32
counter0 ByteString
tag = PhkdfGen
{ phkdfGen_hmacKeyLike :: HmacKeyLike
phkdfGen_hmacKeyLike = HmacKeyLike
key
, phkdfGen_extTag :: ByteString
phkdfGen_extTag = ByteString -> ByteString
extendTag ByteString
tag
, phkdfGen_counter :: Word32
phkdfGen_counter = Word32
counter0
, phkdfGen_state :: ByteString
phkdfGen_state = ByteString
state0
, phkdfGen_initCtx :: Maybe Sha256Ctx
phkdfGen_initCtx = Sha256Ctx -> Maybe Sha256Ctx
forall a. a -> Maybe a
Just Sha256Ctx
ipad0
}
phkdfGen_initHashed :: HmacKeyHashed -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initHashed :: HmacKeyHashed -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initHashed = HmacKeyLike -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initLike (HmacKeyLike -> ByteString -> Word32 -> ByteString -> PhkdfGen)
-> (HmacKeyHashed -> HmacKeyLike)
-> HmacKeyHashed
-> ByteString
-> Word32
-> ByteString
-> PhkdfGen
forall b c a. (b -> c) -> (a -> b) -> a -> c
. HmacKeyHashed -> HmacKeyLike
hmacKeyLike_initHashed
phkdfGen_initPrefixed :: HmacKeyPrefixed -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initPrefixed :: HmacKeyPrefixed -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initPrefixed = HmacKeyLike -> ByteString -> Word32 -> ByteString -> PhkdfGen
phkdfGen_initLike (HmacKeyLike -> ByteString -> Word32 -> ByteString -> PhkdfGen)
-> (HmacKeyPrefixed -> HmacKeyLike)
-> HmacKeyPrefixed
-> ByteString
-> Word32
-> ByteString
-> PhkdfGen
forall b c a. (b -> c) -> (a -> b) -> a -> c
. HmacKeyPrefixed -> HmacKeyLike
hmacKeyLike_initPrefixed
phkdfGen_hmacKeyPlain :: PhkdfGen -> Maybe HmacKeyPlain
phkdfGen_hmacKeyPlain :: PhkdfGen -> Maybe ByteString
phkdfGen_hmacKeyPlain = HmacKeyLike -> Maybe ByteString
hmacKeyLike_toPlain (HmacKeyLike -> Maybe ByteString)
-> (PhkdfGen -> HmacKeyLike) -> PhkdfGen -> Maybe ByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfGen -> HmacKeyLike
phkdfGen_hmacKeyLike
phkdfGen_hmacKeyHashed :: PhkdfGen -> Maybe HmacKeyHashed
phkdfGen_hmacKeyHashed :: PhkdfGen -> Maybe HmacKeyHashed
phkdfGen_hmacKeyHashed = HmacKeyLike -> Maybe HmacKeyHashed
hmacKeyLike_toHashed (HmacKeyLike -> Maybe HmacKeyHashed)
-> (PhkdfGen -> HmacKeyLike) -> PhkdfGen -> Maybe HmacKeyHashed
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfGen -> HmacKeyLike
phkdfGen_hmacKeyLike
phkdfGen_hmacKeyPrefixed :: PhkdfGen -> HmacKeyPrefixed
phkdfGen_hmacKeyPrefixed :: PhkdfGen -> HmacKeyPrefixed
phkdfGen_hmacKeyPrefixed = HmacKeyLike -> HmacKeyPrefixed
hmacKeyLike_toPrefixed (HmacKeyLike -> HmacKeyPrefixed)
-> (PhkdfGen -> HmacKeyLike) -> PhkdfGen -> HmacKeyPrefixed
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfGen -> HmacKeyLike
phkdfGen_hmacKeyLike
phkdfGen_hmacKey :: PhkdfGen -> Maybe HmacKey
phkdfGen_hmacKey :: PhkdfGen -> Maybe HmacKey
phkdfGen_hmacKey = HmacKeyLike -> Maybe HmacKey
hmacKeyLike_toKey (HmacKeyLike -> Maybe HmacKey)
-> (PhkdfGen -> HmacKeyLike) -> PhkdfGen -> Maybe HmacKey
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PhkdfGen -> HmacKeyLike
phkdfGen_hmacKeyLike
phkdfGen_peek :: PhkdfGen -> Maybe ByteString
phkdfGen_peek :: PhkdfGen -> Maybe ByteString
phkdfGen_peek PhkdfGen
gen =
case PhkdfGen -> Maybe Sha256Ctx
phkdfGen_initCtx PhkdfGen
gen of
Maybe Sha256Ctx
Nothing -> ByteString -> Maybe ByteString
forall a. a -> Maybe a
Just (ByteString -> Maybe ByteString) -> ByteString -> Maybe ByteString
forall a b. (a -> b) -> a -> b
$ PhkdfGen -> ByteString
phkdfGen_state PhkdfGen
gen
Just Sha256Ctx
_ -> Maybe ByteString
forall a. Maybe a
Nothing
phkdfGen_toHmacCtx :: PhkdfGen -> HmacCtx
phkdfGen_toHmacCtx :: PhkdfGen -> HmacCtx
phkdfGen_toHmacCtx PhkdfGen
gen =
(HmacKeyLike -> HmacCtx
hmacKeyLike_run (PhkdfGen -> HmacKeyLike
phkdfGen_hmacKeyLike PhkdfGen
gen)) {
hmacCtx_ipadCtx = sha256_update ipad (phkdfGen_state gen)
}
where
ipad :: Sha256Ctx
ipad =
case PhkdfGen -> Maybe Sha256Ctx
phkdfGen_initCtx PhkdfGen
gen of
Maybe Sha256Ctx
Nothing -> HmacCtx -> Sha256Ctx
hmacCtx_ipadCtx (HmacCtx -> Sha256Ctx)
-> (HmacKeyLike -> HmacCtx) -> HmacKeyLike -> Sha256Ctx
forall b c a. (b -> c) -> (a -> b) -> a -> c
. HmacKeyLike -> HmacCtx
hmacKeyLike_run (HmacKeyLike -> Sha256Ctx) -> HmacKeyLike -> Sha256Ctx
forall a b. (a -> b) -> a -> b
$ PhkdfGen -> HmacKeyLike
phkdfGen_hmacKeyLike PhkdfGen
gen
Just Sha256Ctx
x -> Sha256Ctx
x
phkdfGen_head :: PhkdfGen -> ByteString
phkdfGen_head :: PhkdfGen -> ByteString
phkdfGen_head PhkdfGen
gen =
if ByteString -> Int
B.length (PhkdfGen -> ByteString
phkdfGen_extTag PhkdfGen
gen) Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
19 then
PhkdfGen -> HmacCtx
phkdfGen_toHmacCtx PhkdfGen
gen HmacCtx -> (HmacCtx -> HmacCtx) -> HmacCtx
forall a b. a -> (a -> b) -> b
&
[ByteString] -> HmacCtx -> HmacCtx
forall (f :: * -> *).
Foldable f =>
f ByteString -> HmacCtx -> HmacCtx
hmacCtx_feeds [ Word32 -> ByteString
bytestring32 (PhkdfGen -> Word32
phkdfGen_counter PhkdfGen
gen)
, PhkdfGen -> ByteString
phkdfGen_extTag PhkdfGen
gen
] HmacCtx -> (HmacCtx -> ByteString) -> ByteString
forall a b. a -> (a -> b) -> b
&
HmacCtx -> ByteString
hmacCtx_finalize_toByteString
else
PhkdfGen -> HmacCtx
phkdfGen_toHmacCtx PhkdfGen
gen HmacCtx -> (HmacCtx -> HmacCtx) -> HmacCtx
forall a b. a -> (a -> b) -> b
&
[ByteString] -> HmacCtx -> HmacCtx
forall (f :: * -> *).
Foldable f =>
f ByteString -> HmacCtx -> HmacCtx
hmacCtx_feeds [ Word32 -> ByteString
bytestring32 (PhkdfGen -> Word32
phkdfGen_counter PhkdfGen
gen)
, HasCallStack => ByteString -> ByteString
ByteString -> ByteString
B.init (PhkdfGen -> ByteString
phkdfGen_extTag PhkdfGen
gen)
] HmacCtx -> (HmacCtx -> ByteString) -> ByteString
forall a b. a -> (a -> b) -> b
&
ByteString -> Word64 -> HmacCtx -> ByteString
hmacCtx_finalizeBits_toByteString (Word8 -> ByteString
B.singleton (HasCallStack => ByteString -> Word8
ByteString -> Word8
B.last (PhkdfGen -> ByteString
phkdfGen_extTag PhkdfGen
gen))) Word64
7
phkdfGen_read :: PhkdfGen -> (ByteString, PhkdfGen)
phkdfGen_read :: PhkdfGen -> (ByteString, PhkdfGen)
phkdfGen_read PhkdfGen
gen = (ByteString
state', PhkdfGen
gen')
where
state' :: ByteString
state' = PhkdfGen -> ByteString
phkdfGen_head PhkdfGen
gen
gen' :: PhkdfGen
gen' = PhkdfGen
{ phkdfGen_hmacKeyLike :: HmacKeyLike
phkdfGen_hmacKeyLike = PhkdfGen -> HmacKeyLike
phkdfGen_hmacKeyLike PhkdfGen
gen
, phkdfGen_initCtx :: Maybe Sha256Ctx
phkdfGen_initCtx = Maybe Sha256Ctx
forall a. Maybe a
Nothing
, phkdfGen_state :: ByteString
phkdfGen_state = ByteString
state'
, phkdfGen_counter :: Word32
phkdfGen_counter = PhkdfGen -> Word32
phkdfGen_counter PhkdfGen
gen Word32 -> Word32 -> Word32
forall a. Num a => a -> a -> a
+ Word32
1
, phkdfGen_extTag :: ByteString
phkdfGen_extTag = PhkdfGen -> ByteString
phkdfGen_extTag PhkdfGen
gen
}
phkdfGen_toStream :: PhkdfGen -> Stream ByteString
phkdfGen_toStream :: PhkdfGen -> Stream ByteString
phkdfGen_toStream = (PhkdfGen -> (ByteString, PhkdfGen))
-> PhkdfGen -> Stream ByteString
forall c a. (c -> (a, c)) -> c -> Stream a
Stream.unfold PhkdfGen -> (ByteString, PhkdfGen)
phkdfGen_read