/- Copyright (c) 2025 Lean FRO, LLC. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Markus Himmel -/ module prelude public import Init.Data.ByteArray.Basic public import Init.Data.Array.Extract public section -- At present the preferred normal form for empty byte arrays is `ByteArray.empty` @[simp] theorem emptyc_eq_empty : (∅ : ByteArray) = ByteArray.empty := rfl @[simp] theorem emptyWithCapacity_eq_empty : ByteArray.emptyWithCapacity 0 = ByteArray.empty := rfl @[simp] theorem ByteArray.data_empty : ByteArray.empty.data = #[] := rfl @[simp] theorem ByteArray.data_extract {a : ByteArray} {b e : Nat} : (a.extract b e).data = a.data.extract b e := by simp [extract, copySlice] by_cases b ≤ e · rw [(by omega : b + (e - b) = e)] · rw [Array.extract_eq_empty_of_le (by omega), Array.extract_eq_empty_of_le (by omega)] @[simp] theorem ByteArray.extract_zero_size {b : ByteArray} : b.extract 0 b.size = b := by ext1 simp @[simp] theorem ByteArray.extract_same {b : ByteArray} {i : Nat} : b.extract i i = ByteArray.empty := by ext1 simp [Nat.min_le_left] theorem ByteArray.fastAppend_eq_copySlice {a b : ByteArray} : a.fastAppend b = b.copySlice 0 a a.size b.size false := rfl @[simp] theorem List.toByteArray_append {l l' : List UInt8} : (l ++ l').toByteArray = l.toByteArray ++ l'.toByteArray := by simp [List.toByteArray_append'] @[simp] theorem ByteArray.toList_data_append {l l' : ByteArray} : (l ++ l').data.toList = l.data.toList ++ l'.data.toList := by simp [← append_eq] @[simp] theorem ByteArray.data_append {l l' : ByteArray} : (l ++ l').data = l.data ++ l'.data := by simp [← Array.toList_inj] @[simp] theorem ByteArray.size_empty : ByteArray.empty.size = 0 := by simp [← ByteArray.size_data] @[simp] theorem List.data_toByteArray {l : List UInt8} : l.toByteArray.data = l.toArray := by rw [List.toByteArray] suffices ∀ a b, (List.toByteArray.loop a b).data = b.data ++ a.toArray by simpa using this l ByteArray.empty intro a b fun_induction List.toByteArray.loop a b with simp_all @[simp] theorem List.size_toByteArray {l : List UInt8} : l.toByteArray.size = l.length := by simp [← ByteArray.size_data] @[simp] theorem List.toByteArray_nil : List.toByteArray [] = ByteArray.empty := rfl @[simp] theorem ByteArray.empty_append {b : ByteArray} : ByteArray.empty ++ b = b := by ext1 simp @[simp] theorem ByteArray.append_empty {b : ByteArray} : b ++ ByteArray.empty = b := by ext1 simp @[simp, grind =] theorem ByteArray.size_append {a b : ByteArray} : (a ++ b).size = a.size + b.size := by simp [← size_data] @[simp] theorem ByteArray.size_eq_zero_iff {a : ByteArray} : a.size = 0 ↔ a = ByteArray.empty := by refine ⟨fun h => ?_, fun h => h ▸ ByteArray.size_empty⟩ ext1 simp [← Array.size_eq_zero_iff, h] theorem ByteArray.getElem_eq_getElem_data {a : ByteArray} {i : Nat} {h : i < a.size} : a[i] = a.data[i]'(by simpa [← size_data]) := rfl @[simp] theorem ByteArray.getElem_append_left {i : Nat} {a b : ByteArray} {h : i < (a ++ b).size} (hlt : i < a.size) : (a ++ b)[i] = a[i] := by simp only [getElem_eq_getElem_data, data_append] rw [Array.getElem_append_left (by simpa)] theorem ByteArray.getElem_append_right {i : Nat} {a b : ByteArray} {h : i < (a ++ b).size} (hle : a.size ≤ i) : (a ++ b)[i] = b[i - a.size]'(by simp_all; omega) := by simp only [getElem_eq_getElem_data, data_append] rw [Array.getElem_append_right (by simpa)] simp @[simp] theorem List.getElem_toByteArray {l : List UInt8} {i : Nat} {h : i < l.toByteArray.size} : l.toByteArray[i]'h = l[i]'(by simp_all) := by simp [ByteArray.getElem_eq_getElem_data] theorem List.getElem_eq_getElem_toByteArray {l : List UInt8} {i : Nat} {h : i < l.length} : l[i]'h = l.toByteArray[i]'(by simp_all) := by simp @[simp] theorem ByteArray.size_extract {a : ByteArray} {b e : Nat} : (a.extract b e).size = min e a.size - b := by simp [← size_data] @[simp] theorem ByteArray.extract_eq_empty_iff {b : ByteArray} {i j : Nat} : b.extract i j = ByteArray.empty ↔ min j b.size ≤ i := by rw [← size_eq_zero_iff, size_extract] omega @[simp] theorem ByteArray.extract_add_left {b : ByteArray} {i j : Nat} : b.extract (i + j) i = ByteArray.empty := by simp only [extract_eq_empty_iff] exact Nat.le_trans (Nat.min_le_left _ _) (by simp) @[simp] theorem ByteArray.append_eq_empty_iff {a b : ByteArray} : a ++ b = ByteArray.empty ↔ a = ByteArray.empty ∧ b = ByteArray.empty := by simp [← size_eq_zero_iff, size_append] @[simp] theorem List.toByteArray_eq_empty {l : List UInt8} : l.toByteArray = ByteArray.empty ↔ l = [] := by simp [← ByteArray.size_eq_zero_iff] theorem ByteArray.append_right_inj {ys₁ ys₂ : ByteArray} (xs : ByteArray) : xs ++ ys₁ = xs ++ ys₂ ↔ ys₁ = ys₂ := by simp [ByteArray.ext_iff, Array.append_right_inj] @[simp] theorem ByteArray.extract_append_extract {a : ByteArray} {i j k : Nat} : a.extract i j ++ a.extract j k = a.extract (min i j) (max j k) := by ext1 simp theorem ByteArray.extract_eq_extract_append_extract {a : ByteArray} {i k : Nat} (j : Nat) (hi : i ≤ j) (hk : j ≤ k) : a.extract i k = a.extract i j ++ a.extract j k := by simp rw [Nat.min_eq_left hi, Nat.max_eq_right hk] theorem ByteArray.append_inj_left {xs₁ xs₂ ys₁ ys₂ : ByteArray} (h : xs₁ ++ ys₁ = xs₂ ++ ys₂) (hl : xs₁.size = xs₂.size) : xs₁ = xs₂ := by simp only [ByteArray.ext_iff, ← ByteArray.size_data, ByteArray.data_append] at * exact Array.append_inj_left h hl theorem ByteArray.extract_append_eq_right {a b : ByteArray} {i j : Nat} (hi : i = a.size) (hj : j = a.size + b.size) : (a ++ b).extract i j = b := by subst hi hj ext1 simp [← size_data] theorem ByteArray.extract_append_eq_left {a b : ByteArray} {i : Nat} (hi : i = a.size) : (a ++ b).extract 0 i = a := by subst hi ext1 simp theorem ByteArray.extract_append_size_left {a b : ByteArray} {i : Nat} : (a ++ b).extract i a.size = a.extract i a.size := by ext1 simp theorem ByteArray.extract_append_size_add {a b : ByteArray} {i j : Nat} : (a ++ b).extract (a.size + i) (a.size + j) = b.extract i j := by ext1 simp theorem ByteArray.extract_append {as bs : ByteArray} {i j : Nat} : (as ++ bs).extract i j = as.extract i j ++ bs.extract (i - as.size) (j - as.size) := by ext1 simp theorem ByteArray.extract_append_size_add' {a b : ByteArray} {i j k : Nat} (h : k = a.size) : (a ++ b).extract (k + i) (k + j) = b.extract i j := by cases h rw [extract_append_size_add] theorem ByteArray.extract_extract {a : ByteArray} {i j k l : Nat} : (a.extract i j).extract k l = a.extract (i + k) (min (i + l) j) := by ext1 simp theorem ByteArray.getElem_extract_aux {xs : ByteArray} {start stop : Nat} (h : i < (xs.extract start stop).size) : start + i < xs.size := by rw [size_extract] at h; apply Nat.add_lt_of_lt_sub'; apply Nat.lt_of_lt_of_le h apply Nat.sub_le_sub_right; apply Nat.min_le_right theorem ByteArray.getElem_extract {i : Nat} {b : ByteArray} {start stop : Nat} (h) : (b.extract start stop)[i]'h = b[start + i]'(getElem_extract_aux h) := by simp [getElem_eq_getElem_data] theorem ByteArray.extract_eq_extract_left {a : ByteArray} {i i' j : Nat} : a.extract i j = a.extract i' j ↔ min j a.size - i = min j a.size - i' := by simp [ByteArray.ext_iff, Array.extract_eq_extract_left] theorem ByteArray.extract_add_one {a : ByteArray} {i : Nat} (ha : i + 1 ≤ a.size) : a.extract i (i + 1) = [a[i]].toByteArray := by ext · simp omega · rename_i j hj hj' obtain rfl : j = 0 := by simpa using hj' simp [ByteArray.getElem_eq_getElem_data] theorem ByteArray.extract_add_two {a : ByteArray} {i : Nat} (ha : i + 2 ≤ a.size) : a.extract i (i + 2) = [a[i], a[i + 1]].toByteArray := by rw [extract_eq_extract_append_extract (i + 1) (by simp) (by omega), extract_add_one (by omega), extract_add_one (by omega)] simp [← List.toByteArray_append] theorem ByteArray.extract_add_three {a : ByteArray} {i : Nat} (ha : i + 3 ≤ a.size) : a.extract i (i + 3) = [a[i], a[i + 1], a[i + 2]].toByteArray := by rw [extract_eq_extract_append_extract (i + 1) (by simp) (by omega), extract_add_one (by omega), extract_add_two (by omega)] simp [← List.toByteArray_append] theorem ByteArray.extract_add_four {a : ByteArray} {i : Nat} (ha : i + 4 ≤ a.size) : a.extract i (i + 4) = [a[i], a[i + 1], a[i + 2], a[i + 3]].toByteArray := by rw [extract_eq_extract_append_extract (i + 1) (by simp) (by omega), extract_add_one (by omega), extract_add_three (by omega)] simp [← List.toByteArray_append] theorem ByteArray.append_assoc {a b c : ByteArray} : a ++ b ++ c = a ++ (b ++ c) := by ext1 simp @[simp] theorem ByteArray.toList_empty : ByteArray.empty.toList = [] := by simp [ByteArray.toList, ByteArray.toList.loop] theorem ByteArray.copySlice_eq_append {src : ByteArray} {srcOff : Nat} {dest : ByteArray} {destOff len : Nat} {exact : Bool} : ByteArray.copySlice src srcOff dest destOff len exact = dest.extract 0 destOff ++ src.extract srcOff (srcOff +len) ++ dest.extract (destOff + min len (src.data.size - srcOff)) dest.data.size := by ext1 simp [copySlice]