Built with Alectryon, running Coq+SerAPI v8.13.0+0.13.0. Bubbles () indicate interactive fragments: hover for details, tap to reveal contents. Use Ctrl+↑ Ctrl+↓ to navigate, Ctrl+🖱️ to focus. On Mac, use instead of Ctrl.
(* Demo accompanying slides *)

Notation "[ rel _ _ | _ ]" was already used in scope fun_scope. [notation-overridden,parsing]
Notation "[ rel _ _ : _ | _ ]" was already used in scope fun_scope. [notation-overridden,parsing]
Notation "[ rel _ _ in _ & _ | _ ]" was already used in scope fun_scope. [notation-overridden,parsing]
Notation "[ rel _ _ in _ & _ ]" was already used in scope fun_scope. [notation-overridden,parsing]
Notation "[ rel _ _ in _ | _ ]" was already used in scope fun_scope. [notation-overridden,parsing]
Notation "[ rel _ _ in _ ]" was already used in scope fun_scope. [notation-overridden,parsing]
Notation "_ + _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ - _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ <= _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ < _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ >= _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ > _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ <= _ <= _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ < _ <= _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ <= _ < _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ < _ < _" was already used in scope nat_scope. [notation-overridden,parsing]
Notation "_ * _" was already used in scope nat_scope. [notation-overridden,parsing]
Import Monoid.
Record law (T : Type) (idm : T) : Type := Law { operator : T -> T -> T; _ : associative operator; _ : left_id idm operator; _ : right_id idm operator } Arguments law [T]%type_scope _ Arguments Law [T]%type_scope [idm] [operator]%function_scope _ _ _
cat <- operator ( cat_monoid ) div.lcmn <- operator ( lcmn_monoid ) div.gcdn <- operator ( gcdn_monoid ) maxn <- operator ( maxn_monoid ) muln <- operator ( muln_monoid ) addn <- operator ( addn_monoid ) addb <- operator ( addb_monoid ) orb <- operator ( orb_monoid ) andb <- operator ( andb_monoid )
mulmA : forall [T : Type] [idm : T] (mul : law idm), associative mul mulmA is not universe polymorphic Arguments mulmA [T]%type_scope [idm] _ _ _ _ mulmA is transparent Expands to: Constant mathcomp.ssreflect.bigop.Monoid.mulmA
m, n, p, q, r:nat

m + (n + p * (q * r)) = m + n + p * q * r
m, n, p, q, r:nat

m + (n + p * (q * r)) = m + n + p * q * r
by rewrite !mulmA /=. Qed. (* Here is some information which should help you understand how exactly `rewrite mulmA` works when it rewrites in the goal for the first time. mulmA : forall [T : Type] [idm : T] (mul : law idm), associative mul Or, unfolding `associative`: mulmA : forall [T : Type] [idm : T] (mul : law idm) (x y z : T), mul x (mul y z) = mul (mul x y) z Let's define a short-hand: O := @operator The goal (we focus on its left-hand side, LHS): m + (n + p * (q * r)) = ... Unfold notations in the goal: addn m (addn n (muln p (muln q r))) = ... LHS of `mulmA` lemma: mul ?x (mul ?y ?z ) LHS of `mulmA` lemma with inserted coercions: (O ?T ?idm ?mul) ?x ((O ?T ?idm ?mul) ?y ?z ) Let's unify the goal with the LHS of `mulmA` equation: addn m (addn n (muln p (muln q r))) = ... (O ?T ?idm ?mul) ?x ((O ?T ?idm ?mul) ?y ?z ) The head symbol get us addn ≡ @operator ?T ?idm ?mul We have the canonical instance! addn <- operator ( addn_monoid ) Hence, addn ≡ @operator ?T ?idm addn_monoid And `?T` and `?idm` can be deduced from the type of `addn_monoid`: addn ≡ @operator ?T ?idm addn_monoid .. @law ?T ?idm ≡ @law nat 0 So, these unify: addn ≡ operator nat 0 addn_monoid End of inference. This is why after the first rewrite we get the following goal: Lemma foo m n p q r : m + (n + p * (q * r)) = m + n + p * q * r. Proof. rewrite mulmA. Set Printing Coercions. Set Printing Implicit. @operator nat 0 addn_monoid (@operator nat 0 addn_monoid m n) (p * (q * r)) = m + n + p * q * r *)