This PR changes the signature of `Array.set` to take a `Nat`, and a tactic-provided bound, rather than a `Fin`. Corresponding changes (but without the auto-param) for `Array.get` will arrive shortly, after which I'll go more pervasively through the Array API.
75 lines
2.9 KiB
Text
75 lines
2.9 KiB
Text
/-
|
||
Copyright (c) 2022 Microsoft Corporation. All rights reserved.
|
||
Released under Apache 2.0 license as described in the file LICENSE.
|
||
Authors: Leonardo de Moura
|
||
-/
|
||
prelude
|
||
import Init.Data.Array.Basic
|
||
import Init.Data.Nat.Linear
|
||
import Init.NotationExtra
|
||
|
||
theorem Array.of_push_eq_push {as bs : Array α} (h : as.push a = bs.push b) : as = bs ∧ a = b := by
|
||
simp only [push, mk.injEq] at h
|
||
have ⟨h₁, h₂⟩ := List.of_concat_eq_concat h
|
||
cases as; cases bs
|
||
simp_all
|
||
|
||
private theorem List.size_toArrayAux (as : List α) (bs : Array α) : (as.toArrayAux bs).size = as.length + bs.size := by
|
||
induction as generalizing bs with
|
||
| nil => simp [toArrayAux]
|
||
| cons a as ih => simp_arith [toArrayAux, *]
|
||
|
||
private theorem List.of_toArrayAux_eq_toArrayAux {as bs : List α} {cs ds : Array α} (h : as.toArrayAux cs = bs.toArrayAux ds) (hlen : cs.size = ds.size) : as = bs ∧ cs = ds := by
|
||
match as, bs with
|
||
| [], [] => simp [toArrayAux] at h; simp [h]
|
||
| a::as, [] => simp [toArrayAux] at h; rw [← h] at hlen; simp_arith [size_toArrayAux] at hlen
|
||
| [], b::bs => simp [toArrayAux] at h; rw [h] at hlen; simp_arith [size_toArrayAux] at hlen
|
||
| a::as, b::bs =>
|
||
simp [toArrayAux] at h
|
||
have : (cs.push a).size = (ds.push b).size := by simp [*]
|
||
have ⟨ih₁, ih₂⟩ := of_toArrayAux_eq_toArrayAux h this
|
||
simp [ih₁]
|
||
have := Array.of_push_eq_push ih₂
|
||
simp [this]
|
||
|
||
@[simp] theorem List.toArray_eq_toArray_eq (as bs : List α) : (as.toArray = bs.toArray) = (as = bs) := by
|
||
apply propext; apply Iff.intro
|
||
· intro h; simpa [toArray] using h
|
||
· intro h; rw [h]
|
||
|
||
def Array.mapM' [Monad m] (f : α → m β) (as : Array α) : m { bs : Array β // bs.size = as.size } :=
|
||
go 0 ⟨mkEmpty as.size, rfl⟩ (by simp)
|
||
where
|
||
go (i : Nat) (acc : { bs : Array β // bs.size = i }) (hle : i ≤ as.size) : m { bs : Array β // bs.size = as.size } := do
|
||
if h : i = as.size then
|
||
return h ▸ acc
|
||
else
|
||
have hlt : i < as.size := Nat.lt_of_le_of_ne hle h
|
||
let b ← f as[i]
|
||
go (i+1) ⟨acc.val.push b, by simp [acc.property]⟩ hlt
|
||
termination_by as.size - i
|
||
decreasing_by decreasing_trivial_pre_omega
|
||
|
||
@[inline] private unsafe def mapMonoMImp [Monad m] (as : Array α) (f : α → m α) : m (Array α) :=
|
||
go 0 as
|
||
where
|
||
@[specialize] go (i : Nat) (as : Array α) : m (Array α) := do
|
||
if h : i < as.size then
|
||
let a := as[i]
|
||
let b ← f a
|
||
if ptrEq a b then
|
||
go (i+1) as
|
||
else
|
||
go (i+1) (as.set i b h)
|
||
else
|
||
return as
|
||
|
||
/--
|
||
Monomorphic `Array.mapM`. The internal implementation uses pointer equality, and does not allocate a new array
|
||
if the result of each `f a` is a pointer equal value `a`.
|
||
-/
|
||
@[implemented_by mapMonoMImp] def Array.mapMonoM [Monad m] (as : Array α) (f : α → m α) : m (Array α) :=
|
||
as.mapM f
|
||
|
||
@[inline] def Array.mapMono (as : Array α) (f : α → α) : Array α :=
|
||
Id.run <| as.mapMonoM f
|