aboutsummaryrefslogtreecommitdiff
path: root/mathcomp/real_closed/polyrcf.v
diff options
context:
space:
mode:
authorCyril Cohen2018-04-17 17:00:15 +0200
committerCyril Cohen2018-04-18 10:49:18 +0200
commit13479422b0286c86d0888e06aba112153ca6314d (patch)
tree6b921cad503e12fcea8dc7cc136667a54ea86bf4 /mathcomp/real_closed/polyrcf.v
parentc1ec9cd8e7e50f73159613c492aad4c6c40bc3aa (diff)
Moving real_closed to another repo
Diffstat (limited to 'mathcomp/real_closed/polyrcf.v')
-rw-r--r--mathcomp/real_closed/polyrcf.v1811
1 files changed, 0 insertions, 1811 deletions
diff --git a/mathcomp/real_closed/polyrcf.v b/mathcomp/real_closed/polyrcf.v
deleted file mode 100644
index 8aaeb97..0000000
--- a/mathcomp/real_closed/polyrcf.v
+++ /dev/null
@@ -1,1811 +0,0 @@
-(* (c) Copyright 2006-2016 Microsoft Corporation and Inria. *)
-(* Distributed under the terms of CeCILL-B. *)
-Require Import mathcomp.ssreflect.ssreflect.
-From mathcomp
-Require Import ssrfun ssrbool eqtype ssrnat seq choice fintype.
-From mathcomp
-Require Import bigop ssralg poly polydiv ssrnum zmodp.
-From mathcomp
-Require Import polyorder path interval ssrint.
-
-(****************************************************************************)
-(* This files contains basic (and unformatted) theory for polynomials *)
-(* over a realclosed fields. From the IVT (contained in the rcfType *)
-(* structure), we derive Rolle's Theorem, the Mean Value Theorem, a root *)
-(* isolation procedure and the notion of neighborhood. *)
-(* *)
-(* sgp_minfty p == the sign of p in -oo *)
-(* sgp_pinfty p == the sign of p in +oo *)
-(* cauchy_bound p == the cauchy bound of p *)
-(* (this strictly bounds the norm of roots of p) *)
-(* roots p a b == the ordered list of roots of p in `[a, b] *)
-(* defaults to [::] when p == 0 *)
-(* rootsR p == the ordered list of all roots of p, ([::] if p == 0). *)
-(* next_root p x b == the smallest root of p contained in `[x, maxn x b] *)
-(* if p has no root on `[x, maxn x b], we pick maxn x b. *)
-(* prev_root p x a == the smallest root of p contained in `[minn x a, x] *)
-(* if p has no root on `[minn x a, x], we pick minn x a. *)
-(* neighpr p a b := `]a, next_root p a b[. *)
-(* == an open interval of the form `]a, x[, with x <= b *)
-(* in which p has no root. *)
-(* neighpl p a b := `]prev_root p a b, b[. *)
-(* == an open interval of the form `]x, b[, with a <= x *)
-(* in which p has no root. *)
-(* sgp_right p a == the sign of p on the right of a. *)
-(****************************************************************************)
-
-
-Import GRing.Theory Num.Theory Num.Def.
-Import Pdiv.Idomain.
-
-Set Implicit Arguments.
-Unset Strict Implicit.
-Unset Printing Implicit Defensive.
-
-Local Open Scope nat_scope.
-Local Open Scope ring_scope.
-
-Local Notation noroot p := (forall x, ~~ root p x).
-Local Notation mid x y := ((x + y) / 2%:R).
-
-Section more.
-Section SeqR.
-
-Lemma last1_neq0 (R : ringType) (s : seq R) (c : R) :
- c != 0 -> (last c s != 0) = (last 1 s != 0).
-Proof. by elim: s c => [|t s ihs] c cn0 //; rewrite oner_eq0 cn0. Qed.
-
-End SeqR.
-
-Section poly.
-Import Pdiv.Ring Pdiv.ComRing.
-
-Variable R : idomainType.
-
-Implicit Types p q : {poly R}.
-
-Lemma lead_coefDr p q :
- (size q > size p)%N -> lead_coef (p + q) = lead_coef q.
-Proof. by move/lead_coefDl<-; rewrite addrC. Qed.
-
-Lemma leq1_size_polyC (c : R) : (size c%:P <= 1)%N.
-Proof. by rewrite size_polyC; case: (c == 0). Qed.
-
-Lemma my_size_exp p n :
- p != 0 -> (size (p ^+ n)) = ((size p).-1 * n).+1%N.
-Proof.
-by move=> hp; rewrite -size_exp prednK // lt0n size_poly_eq0 expf_neq0.
-Qed.
-
-Lemma coef_comp_poly p q n :
- (p \Po q)`_n = \sum_(i < size p) p`_i * (q ^+ i)`_n.
-Proof.
-rewrite comp_polyE coef_sum.
-by elim/big_ind2: _ => [//|? ? ? ? -> -> //|i]; rewrite coefZ.
-Qed.
-
-Lemma gt_size_poly p n : (size p > n)%N -> p != 0.
-Proof.
-by move=> h; rewrite -size_poly_eq0 lt0n_neq0 //; apply: leq_ltn_trans h.
-Qed.
-
-Lemma lead_coef_comp_poly p q :
- (size q > 1)%N ->
- lead_coef (p \Po q) = (lead_coef p) * (lead_coef q) ^+ (size p).-1.
-Proof.
-move=> sq; rewrite !lead_coefE coef_comp_poly size_comp_poly.
-case hp: (size p) => [|n].
- move/eqP: hp; rewrite size_poly_eq0 => /eqP ->.
- by rewrite big_ord0 coef0 mul0r.
-rewrite big_ord_recr /= big1 => [|i _].
- by rewrite add0r -lead_coefE -lead_coef_exp lead_coefE size_exp mulnC.
-rewrite [X in _ * X]nth_default ?mulr0 ?(leq_trans (size_exp_leq _ _)) //.
-by rewrite mulnC ltn_mul2r -subn1 subn_gt0 sq /=.
-Qed.
-
-End poly.
-End more.
-
-(******************************************************************)
-(* Definitions and properties for polynomials in a numDomainType. *)
-(******************************************************************)
-Section PolyNumDomain.
-
-Variable R : numDomainType.
-Implicit Types (p q : {poly R}).
-
-Definition sgp_pinfty (p : {poly R}) := sgr (lead_coef p).
-Definition sgp_minfty (p : {poly R}) :=
- sgr ((-1) ^+ (size p).-1 * (lead_coef p)).
-
-End PolyNumDomain.
-
-(******************************************************************)
-(* Definitions and properties for polynomials in a realFieldType. *)
-(******************************************************************)
-Section PolyRealField.
-
-Variable R : realFieldType.
-Implicit Types (p q : {poly R}).
-
-Section SgpInfty.
-
-Lemma sgp_pinfty_sym p : sgp_pinfty (p \Po -'X) = sgp_minfty p.
-Proof.
-rewrite /sgp_pinfty /sgp_minfty lead_coef_comp_poly ?size_opp ?size_polyX //.
-by rewrite lead_coef_opp lead_coefX mulrC.
-Qed.
-
-Lemma poly_pinfty_gt_lc p :
- lead_coef p > 0 ->
- exists n, forall x, x >= n -> p.[x] >= lead_coef p.
-Proof.
-elim/poly_ind: p => [| q c IHq]; first by rewrite lead_coef0 ltrr.
-have [->|q_neq0] := eqVneq q 0.
- by rewrite mul0r add0r lead_coefC => c_gt0; exists 0 => x _; rewrite hornerC.
-rewrite lead_coefDl ?size_mul ?polyX_eq0 // ?lead_coefMX; last first.
- rewrite size_polyX addn2 size_polyC /= ltnS.
- by rewrite (leq_trans (leq_b1 _)) // size_poly_gt0.
-move=> lq_gt0; have [y Hy] := IHq lq_gt0.
-pose z := (1 + (lead_coef q) ^-1 * `|c|); exists (maxr y z) => x.
-have z_gt0 : 0 < z by rewrite ltr_spaddl ?ltr01 ?mulr_ge0 ?invr_ge0 // ltrW.
-rewrite !hornerE ler_maxl => /andP[/Hy Hq Hc].
-rewrite (@ler_trans _ (lead_coef q * z + c)) //; last first.
- rewrite ler_add2r (@ler_trans _ (q.[x] * z)) // ?ler_pmul2r //.
- by rewrite ler_pmul2l // (ltr_le_trans _ Hq).
-rewrite mulrDr mulr1 -addrA ler_addl mulVKf ?gtr_eqF //.
-by rewrite -[c]opprK subr_ge0 normrN ler_norm.
-Qed.
-
-(* :REMARK: not necessary here ! *)
-Lemma poly_lim_infty p m :
- lead_coef p > 0 -> (size p > 1)%N ->
- exists n, forall x, x >= n -> p.[x] >= m.
-Proof.
-elim/poly_ind: p m => [| q c _] m; first by rewrite lead_coef0 ltrr.
-have [-> _|q_neq0] := eqVneq q 0.
- by rewrite mul0r add0r size_polyC ltnNge leq_b1.
-rewrite lead_coefDl ?size_mul ?polyX_eq0 // ?lead_coefMX; last first.
- rewrite size_polyX addn2 size_polyC /= ltnS.
- by rewrite (leq_trans (leq_b1 _)) // size_poly_gt0.
-move=> lq_gt0; have [y Hy _] := poly_pinfty_gt_lc lq_gt0.
-pose z := (1 + (lead_coef q) ^-1 * (`|m| + `|c|)); exists (maxr y z) => x.
-have z_gt0 : 0 < z.
- by rewrite ltr_spaddl ?ltr01 ?mulr_ge0 ?invr_ge0 ?addr_ge0 // ?ltrW.
-rewrite !hornerE ler_maxl => /andP[/Hy Hq Hc].
-rewrite (@ler_trans _ (lead_coef q * z + c)) //; last first.
- rewrite ler_add2r (@ler_trans _ (q.[x] * z)) // ?ler_pmul2r //.
- by rewrite ler_pmul2l // (ltr_le_trans _ Hq).
-rewrite mulrDr mulr1 mulVKf ?gtr_eqF // addrA -addrA ler_paddr //.
- by rewrite -[c]opprK subr_ge0 normrN ler_norm.
-by rewrite ler_paddl ?ler_norm // ?ltrW.
-Qed.
-
-End SgpInfty.
-
-Section CauchyBound.
-
-Definition cauchy_bound (p : {poly R}) :=
- 1 + `|lead_coef p|^-1 * \sum_(i < size p) `|p`_i|.
-
-(* Could be a sharp bound, and proof should shrink... *)
-Lemma cauchy_boundP (p : {poly R}) x :
- p != 0 -> p.[x] = 0 -> `| x | < cauchy_bound p.
-Proof.
-move=> np0 rpx; rewrite ltr_spaddl ?ltr01 //.
-case e: (size p) => [|n]; first by move: np0; rewrite -size_poly_eq0 e eqxx.
-have lcp : `|lead_coef p| > 0 by move: np0; rewrite -lead_coef_eq0 -normr_gt0.
-have lcn0 : `|lead_coef p| != 0 by rewrite normr_eq0 -normr_gt0.
-case: (lerP `|x| 1) => cx1.
- rewrite (ler_trans cx1) // /cauchy_bound ler_pdivl_mull // mulr1.
- by rewrite big_ord_recr /= /lead_coef e ler_addr sumr_ge0.
-case es: n e => [|m] e.
- suff p0 : p = 0 by rewrite p0 eqxx in np0.
- by move: rpx; rewrite (@size1_polyC _ p) ?e ?lerr // hornerC; move->.
-move: rpx; rewrite horner_coef e -es big_ord_recr /=; move/eqP; rewrite eq_sym.
-rewrite -subr_eq sub0r; move/eqP => h1.
-have {h1} h1 : `|p`_n| * `|x| ^+ n <= \sum_(i < n) `|p`_i * x ^+ i|.
- rewrite -normrX -normrM -normrN h1.
- by rewrite (ler_trans (ler_norm_sum _ _ _)) // lerr.
-have xp : `| x | > 0 by rewrite (ltr_trans _ cx1) ?ltr01.
-move: h1; rewrite {-1}es exprS mulrA -ler_pdivl_mulr ?exprn_gt0 //.
-rewrite big_distrl /= big_ord_recr /= normrM normrX -mulrA es mulfV; last first.
- by rewrite expf_eq0 negb_and eq_sym (ltr_eqF xp) orbT.
-have pnp : 0 < `|p`_n| by move: lcp; rewrite /lead_coef e es.
-rewrite mulr1 -es mulrC -ler_pdivl_mulr //.
-rewrite [_ / _]mulrC /cauchy_bound /lead_coef e -es /=.
-move=> h1; apply: (ler_trans h1) => //.
-rewrite ler_pmul2l ?invr_gt0 ?(ltrW pnp) // big_ord_recr /=.
-rewrite es [_ + `|p`_m.+1|]addrC ler_paddl // ?normr_ge0 //.
-rewrite big_ord_recr /= ler_add2r; apply: ler_sum => i.
-rewrite normrM normrX.
-rewrite -mulrA ler_pimulr ?normrE // ler_pdivr_mulr ?exprn_gt0 // mul1r.
-by rewrite ler_eexpn2l // 1?ltrW //; case: i=> i hi /=; rewrite ltnW.
-(* this could be improved a little bit with int exponents *)
-Qed.
-
-Lemma le_cauchy_bound p : p != 0 -> {in `]-oo, (- cauchy_bound p)], noroot p}.
-Proof.
-move=> p_neq0 x; rewrite inE /= lerNgt; apply: contra => /rootP.
-by move=> /(cauchy_boundP p_neq0) /ltr_normlP []; rewrite ltr_oppl.
-Qed.
-Hint Resolve le_cauchy_bound.
-
-Lemma ge_cauchy_bound p : p != 0 -> {in `[cauchy_bound p, +oo[, noroot p}.
-Proof.
-move=> p_neq0 x; rewrite inE andbT /= lerNgt; apply: contra => /rootP.
-by move=> /(cauchy_boundP p_neq0) /ltr_normlP []; rewrite ltr_oppl.
-Qed.
-Hint Resolve ge_cauchy_bound.
-
-Lemma cauchy_bound_gt0 p : cauchy_bound p > 0.
-Proof.
-rewrite ltr_spaddl ?ltr01 ?mulr_ge0 ?invr_ge0 ?normr_ge0 //.
-by rewrite sumr_ge0 // => i; rewrite normr_ge0.
-Qed.
-Hint Resolve cauchy_bound_gt0.
-
-Lemma cauchy_bound_ge0 p : cauchy_bound p >= 0.
-Proof. by rewrite ltrW. Qed.
-Hint Resolve cauchy_bound_ge0.
-
-End CauchyBound.
-
-End PolyRealField.
-
-(************************************************************)
-(* Definitions and properties for polynomials in a rcfType. *)
-(************************************************************)
-Section PolyRCF.
-
-Variable R : rcfType.
-
-Section Prelim.
-
-Implicit Types a b c : R.
-Implicit Types x y z t : R.
-Implicit Types p q r : {poly R}.
-
-(* we restate poly_ivt in a nicer way. Perhaps the def of PolyRCF should *)
-(* be moved in this file, juste above this section *)
-
-Lemma poly_ivt (p : {poly R}) (a b : R) :
- a <= b -> 0 \in `[p.[a], p.[b]] -> { x : R | x \in `[a, b] & root p x }.
-Proof. by move=> leab root_p_ab; apply/sig2W/poly_ivt. Qed.
-
-Lemma polyrN0_itv (i : interval R) (p : {poly R}) :
- {in i, noroot p} ->
- forall y x : R, y \in i -> x \in i -> sgr p.[x] = sgr p.[y].
-Proof.
-move=> hi y x hy hx; wlog xy: x y hx hy / x <= y=> [hwlog|].
- by case/orP: (ler_total x y)=> xy; [|symmetry]; apply: hwlog.
-have hxyi: {subset `[x, y] <= i}.
- move=> z; apply: subitvP=> /=.
- by case: i hx hy {hi}=> [[[] ?|] [[] ?|]] /=; do ?[move/itvP->|move=> ?].
-do 2![case: sgrP; first by move/rootP; rewrite (negPf (hi _ _))]=> //.
- move=> /ltrW py0 /ltrW p0x; case: (@poly_ivt (- p) x y)=> //.
- by rewrite inE !hornerN !oppr_cp0 p0x.
- by move=> z hz; rewrite rootN (negPf (hi z _)) // hxyi.
-move=> /ltrW p0y /ltrW px0; case: (@poly_ivt p x y); rewrite ?inE ?px0 //.
-by move=> z hz; rewrite (negPf (hi z _)) // hxyi.
-Qed.
-
-Lemma poly_div_factor (a : R) (P : {poly R} -> Prop) :
- (forall k, P k%:P) ->
- (forall p n k, p.[a] != 0 -> P p -> P (p * ('X - a%:P)^+(n.+1) + k%:P)%R)
- -> forall p, P p.
-Proof.
-move=> Pk Pq p.
-move: {-2}p (leqnn (size p)); elim: (size p)=> {p} [|n ihn] p spn.
- move: spn; rewrite leqn0 size_poly_eq0; move/eqP->; rewrite -polyC0.
- exact: Pk.
-case: (leqP (size p) 1)=> sp1; first by rewrite [p]size1_polyC ?sp1//.
-rewrite (Pdiv.IdomainMonic.divp_eq (monicXsubC a) p).
-rewrite [_ %% _]size1_polyC; last first.
- rewrite -ltnS.
- by rewrite (@leq_trans (size ('X - a%:P))) //
- ?ltn_modp ?polyXsubC_eq0 ?size_XsubC.
-have [n' [q hqa hp]] := multiplicity_XsubC (p %/ ('X - a%:P)) a.
-rewrite divpN0 ?size_XsubC ?polyXsubC_eq0 ?sp1 //= in hqa.
-rewrite hp -mulrA -exprSr; apply: Pq=> //; apply: ihn.
-rewrite (@leq_trans (size (q * ('X - a%:P) ^+ n'))) //.
- rewrite size_mul ?expf_eq0 ?polyXsubC_eq0 ?andbF //; last first.
- by apply: contra hqa; move/eqP->; rewrite root0.
- by rewrite size_exp_XsubC addnS leq_addr.
-by rewrite -hp size_divp ?polyXsubC_eq0 ?size_XsubC // leq_subLR.
-Qed.
-
-Lemma nth_root x n : x > 0 -> { y | y > 0 & y ^+ (n.+1) = x }.
-Proof.
-move=> l0x.
-case: (ltrgtP x 1)=> hx; last by exists 1; rewrite ?hx ?lter01// expr1n.
- case: (@poly_ivt ('X ^+ n.+1 - x%:P) 0 1); first by rewrite ler01.
- rewrite ?(hornerE,horner_exp) ?inE.
- by rewrite exprS mul0r sub0r expr1n oppr_cp0 subr_gte0/= !ltrW.
- move=> y; case/andP=> [l0y ly1]; rewrite rootE ?(hornerE,horner_exp).
- rewrite subr_eq0; move/eqP=> hyx; exists y=> //; rewrite lt0r l0y.
- rewrite andbT; apply/eqP=> y0; move: hyx; rewrite y0.
- by rewrite exprS mul0r=> x0; move: l0x; rewrite -x0 ltrr.
-case: (@poly_ivt ('X ^+ n.+1 - x%:P) 0 x); first by rewrite ltrW.
- rewrite ?(hornerE,horner_exp) exprS mul0r sub0r ?inE.
- by rewrite oppr_cp0 (ltrW l0x) subr_ge0 ler_eexpr // ltrW.
-move=> y; case/andP=> l0y lyx; rewrite rootE ?(hornerE,horner_exp).
-rewrite subr_eq0; move/eqP=> hyx; exists y=> //; rewrite lt0r l0y.
-rewrite andbT; apply/eqP=> y0; move: hyx; rewrite y0.
-by rewrite exprS mul0r=> x0; move: l0x; rewrite -x0 ltrr.
-Qed.
-
-Lemma poly_cont x p e :
- e > 0 -> exists2 d, d > 0 & forall y, `|y - x| < d -> `|p.[y] - p.[x]| < e.
-Proof.
-elim/(@poly_div_factor x): p e.
- move=> e ep; exists 1; rewrite ?ltr01// => y hy.
- by rewrite !hornerC subrr normr0.
-move=> p n k pxn0 Pp e ep.
-case: (Pp (`|p.[x]|/2%:R)).
- by rewrite pmulr_lgt0 ?invr_gte0//= ?ltr0Sn// normrE.
-move=> d' d'p He'.
-case: (@nth_root (e / ((3%:R / 2%:R) * `|p.[x]|)) n).
- by rewrite ltr_pdivl_mulr ?mul0r ?pmulr_rgt0 ?invr_gt0 ?normrE ?ltr0Sn.
-move=> d dp rootd.
-exists (minr d d'); first by rewrite ltr_minr dp.
-move=> y; rewrite ltr_minr; case/andP=> hxye hxye'.
-rewrite !(hornerE, horner_exp) subrr [0 ^+ _]exprS mul0r mulr0 add0r addrK.
-rewrite normrM (@ler_lt_trans _ (`|p.[y]| * d ^+ n.+1)) //.
- by rewrite ler_wpmul2l ?normrE // normrX ler_expn2r -?topredE /= ?normrE 1?ltrW.
-rewrite rootd mulrCA gtr_pmulr //.
-rewrite ltr_pdivr_mulr ?mul1r ?pmulr_rgt0 ?invr_gt0 ?ltr0Sn ?normrE //.
-rewrite mulrDl mulrDl divff; last by rewrite -mulr2n pnatr_eq0.
-rewrite !mul1r mulrC -ltr_subl_addr.
-by rewrite (ler_lt_trans _ (He' y _)) // ler_sub_dist.
-Qed.
-
-Lemma poly_ltsp_roots p (rs : seq R) :
- (size rs >= size p)%N -> uniq rs -> all (root p) rs -> p = 0.
-Proof.
-move=> hrs urs rrs; apply/eqP; apply: contraLR hrs=> np0.
-by rewrite -ltnNge; apply: max_poly_roots.
-Qed.
-
-Lemma ivt_sign (p : {poly R}) (a b : R) :
- a <= b -> sgr p.[a] * sgr p.[b] = -1 -> { x : R | x \in `]a, b[ & root p x}.
-Proof.
-move=> hab /eqP; rewrite mulrC mulr_sg_eqN1=> /andP [spb0 /eqP spb].
- case: (@poly_ivt (sgr p.[b] *: p) a b)=> //.
- by rewrite !hornerZ {1}spb mulNr -!normrEsg inE /= oppr_cp0 !normrE.
-move=> c hc; rewrite rootZ ?sgr_eq0 // => rpc; exists c=> //.
-(* need for a lemma reditvP *)
-rewrite inE /= !ltr_neqAle andbCA -!andbA [_ && (_ <= _)]hc andbT.
-rewrite eq_sym -negb_or.
-apply/negP=> /orP [] /eqP ec; move: rpc; rewrite -ec /root ?(negPf spb0) //.
-by rewrite -sgr_cp0 -[sgr _]opprK -spb eqr_oppLR oppr0 sgr_cp0 (negPf spb0).
-Qed.
-
-Let rolle_weak a b p :
- a < b -> p.[a] = 0 -> p.[b] = 0 ->
- {c | (c \in `]a, b[) & (p^`().[c] == 0) || (p.[c] == 0)}.
-Proof.
-move=> lab pa0 pb0; apply/sig2W.
-case p0: (p == 0).
- rewrite (eqP p0); exists (mid a b); first by rewrite mid_in_itv.
- by rewrite derivC horner0 eqxx.
-have [n [p' p'a0 hp]] := multiplicity_XsubC p a; rewrite p0 /= in p'a0.
-case: n hp pa0 p0 pb0 p'a0=> [ | n -> _ p0 pb0 p'a0].
- by rewrite {1}expr0 mulr1 rootE=> ->; move/eqP->.
-have [m [q qb0 hp']] := multiplicity_XsubC p' b.
-rewrite (contraNneq _ p'a0) /= in qb0 => [|->]; last exact: root0.
-case: m hp' pb0 p0 p'a0 qb0=> [|m].
- rewrite {1}expr0 mulr1=> ->; move/eqP.
- rewrite !(hornerE, horner_exp, mulf_eq0).
- by rewrite !expf_eq0 !subr_eq0 !(gtr_eqF lab) !andbF !orbF !rootE=> ->.
-move=> -> _ p0 p'a0 qb0; case: (sgrP (q.[a] * q.[b])); first 2 last.
-- move=> sqasb; case: (@ivt_sign q a b)=> //; first exact: ltrW.
- by apply/eqP; rewrite -sgrM sgr_cp0.
- move=> c lacb rqc; exists c=> //.
- by rewrite !hornerM (eqP rqc) !mul0r eqxx orbT.
-- move/eqP; rewrite mulf_eq0 (rootPf qb0) orbF; move/eqP=> qa0.
- by move: p'a0; rewrite ?rootM rootE qa0 eqxx.
-- move=> hspq; rewrite !derivCE /= !mul1r mulrDl !pmulrn.
- set xan := (('X - a%:P) ^+ n); set xbm := (('X - b%:P) ^+ m).
- have ->: ('X - a%:P) ^+ n.+1 = ('X - a%:P) * xan by rewrite exprS.
- have ->: ('X - b%:P) ^+ m.+1 = ('X - b%:P) * xbm by rewrite exprS.
- rewrite -mulrzl -[_ *~ n.+1]mulrzl.
- have fac : forall x y z : {poly R}, x * (y * xbm) * (z * xan)
- = (y * z * x) * (xbm * xan).
- by move=> x y z; rewrite mulrCA !mulrA [_ * y]mulrC mulrA.
- rewrite !fac -!mulrDl; set r := _ + _ + _.
- case: (@ivt_sign (sgr q.[b] *: r) a b); first exact: ltrW.
- rewrite !hornerZ !sgr_smul mulrACA -expr2 sqr_sg (rootPf qb0) mul1r.
- rewrite !(subrr, mul0r, mulr0, addr0, add0r, hornerC, hornerXsubC,
- hornerD, hornerN, hornerM, hornerMn).
- rewrite [_ * _%:R]mulrC -!mulrA !pmulrn !mulrzl !sgrMz -sgrM.
- rewrite mulrAC mulrA -mulrA sgrM -opprB mulNr sgrN sgrM.
- by rewrite !gtr0_sg ?subr_gt0 ?mulr1 // mulrC.
-move=> c lacb; rewrite rootE hornerZ mulf_eq0.
-rewrite sgr_cp0 (rootPf qb0) orFb=> rc0.
-by exists c=> //; rewrite !hornerM !mulf_eq0 rc0.
-Qed.
-
-Theorem rolle a b p :
- a < b -> p.[a] = p.[b] -> {c | c \in `]a, b[ & p^`().[c] = 0}.
-Proof.
-move=> lab pab.
-wlog pb0 : p pab / p.[b] = 0 => [hwlog|].
- case: (hwlog (p - p.[b]%:P)); rewrite ?hornerE ?pab ?subrr //.
- by move=> c acb; rewrite derivE derivC subr0=> hc; exists c.
-move: pab; rewrite pb0=> pa0.
-have: (forall rs : seq R, {subset rs <= `]a, b[} ->
- (size p <= size rs)%N -> uniq rs -> all (root p) rs -> p = 0).
- by move=> rs hrs; apply: poly_ltsp_roots.
-elim: (size p) a b lab pa0 pb0=> [|n ihn] a b lab pa0 pb0 max_roots.
- rewrite (@max_roots [::]) //=.
- by exists (mid a b); rewrite ?mid_in_itv // derivE horner0.
-case: (@rolle_weak a b p); rewrite // ?pa0 ?pb0 //=.
-move=> c hc; case: (altP (_ =P 0))=> //= p'c0 pc0; first by exists c.
-suff: { d : R | d \in `]a, c[ & (p^`()).[d] = 0 }.
- case=> [d hd] p'd0; exists d=> //.
- by apply: subitvPr hd; rewrite //= (itvP hc).
-apply: ihn=> //; first by rewrite (itvP hc).
- exact/eqP.
-move=> rs hrs srs urs rrs; apply: (max_roots (c :: rs))=> //=; last exact/andP.
- move=> x; rewrite in_cons; case/predU1P=> hx; first by rewrite hx.
- have: x \in `]a, c[ by apply: hrs.
- by apply: subitvPr; rewrite /= (itvP hc).
-by rewrite urs andbT; apply/negP; move/hrs; rewrite bound_in_itv.
-Qed.
-
-Theorem mvt a b p :
- a < b -> {c | c \in `]a, b[ & p.[b] - p.[a] = p^`().[c] * (b - a)}.
-Proof.
-move=> lab.
-pose q := (p.[b] - p.[a])%:P * ('X - a%:P) - (b - a)%:P * (p - p.[a]%:P).
-case: (@rolle a b q)=> //.
- by rewrite /q !hornerE !(subrr,mulr0) mulrC subrr.
-move=> c lacb q'x0; exists c=> //.
-move: q'x0; rewrite /q !derivE !(mul0r,add0r,subr0,mulr1).
-by move/eqP; rewrite !hornerE mulrC subr_eq0; move/eqP.
-Qed.
-
-Lemma deriv_sign a b p :
- (forall x, x \in `]a, b[ -> p^`().[x] >= 0)
- -> (forall x y, (x \in `]a, b[) && (y \in `]a, b[)
- -> x < y -> p.[x] <= p.[y] ).
-Proof.
-move=> Pab x y; case/andP=> hx hy xy.
-rewrite -subr_gte0; case: (@mvt x y p)=> //.
-move=> c hc ->; rewrite pmulr_lge0 ?subr_gt0 ?Pab //.
-by apply: subitvP hc; rewrite //= ?(itvP hx) ?(itvP hy).
-Qed.
-
-End Prelim.
-
-Section MonotonictyAndRoots.
-
-Section NoRoot.
-
-Variable (p : {poly R}).
-
-Variables (a b : R).
-
-Hypothesis der_pos : forall x, x \in `]a, b[ -> (p^`()).[x] > 0.
-
-Lemma derp0r : 0 <= p.[a] -> forall x, x \in `]a, b] -> p.[x] > 0.
-Proof.
-move=> pa0 x; case/itv_dec=> ax xb; case: (mvt p ax) => c acx.
-move/(canRL (@subrK _ _))->; rewrite ltr_paddr //.
-by rewrite pmulr_rgt0 ?subr_gt0 // der_pos //; apply: subitvPr acx.
-Qed.
-
-Lemma derppr : 0 < p.[a] -> forall x, x \in `[a, b] -> p.[x] > 0.
-Proof.
-move=> pa0 x hx; case exa: (x == a); first by rewrite (eqP exa).
-case: (@mvt a x p); first by rewrite ltr_def exa (itvP hx).
-move=> c hc; move/eqP; rewrite subr_eq; move/eqP->; rewrite ltr_spsaddr //.
-rewrite pmulr_rgt0 ?subr_gt0 //; first by rewrite ltr_def exa (itvP hx).
-by rewrite der_pos // (subitvPr _ hc) //= ?(itvP hx).
-Qed.
-
-Lemma derp0l : p.[b] <= 0 -> forall x, x \in `[a, b[ -> p.[x] < 0.
-Proof.
-move=> pb0 x hx; rewrite -oppr_gte0 /=.
-case: (@mvt x b p)=> //; first by rewrite (itvP hx).
-move=> c hc; move/(canRL (@addKr _ _))->; rewrite ltr_spaddr ?oppr_ge0 //.
-rewrite pmulr_rgt0 // ?subr_gt0 ?(itvP hx) //.
-by rewrite der_pos // (subitvPl _ hc) //= (itvP hx).
-Qed.
-
-Lemma derpnl : p.[b] < 0 -> forall x, x \in `[a, b] -> p.[x] < 0.
-Proof.
-move=> pbn x hx; case xb: (b == x) pbn; first by rewrite -(eqP xb).
-case: (@mvt x b p); first by rewrite ltr_def xb ?(itvP hx).
-move=> y hy; move/eqP; rewrite subr_eq; move/eqP->.
-rewrite !ltrNge; apply: contra=> hpx; rewrite ler_paddr // ltrW //.
-rewrite pmulr_rgt0 ?subr_gt0 ?(itvP hy) //.
-by rewrite der_pos // (subitvPl _ hy) //= (itvP hx).
-Qed.
-
-End NoRoot.
-
-Section NoRoot_sg.
-
-Variable (p : {poly R}).
-
-Variables (a b c : R).
-
-Hypothesis derp_neq0 : {in `]a, b[, noroot p^`()}.
-Hypothesis acb : c \in `]a, b[.
-
-Local Notation sp'c := (sgr p^`().[c]).
-Local Notation q := (sp'c *: p).
-
-Fact derq_pos x : x \in `]a, b[ -> (q^`()).[x] > 0.
-Proof.
-move=> hx; rewrite derivZ hornerZ -sgr_cp0.
-rewrite sgrM sgr_id mulr_sg_eq1 derp_neq0 //=.
-by apply/eqP; apply: (@polyrN0_itv `]a, b[).
-Qed.
-
-Fact sgp x : sgr p.[x] = sp'c * sgr q.[x].
-Proof.
-by rewrite hornerZ sgr_smul mulrA -expr2 sqr_sg derp_neq0 ?mul1r.
-Qed.
-
-Fact hsgp x : 0 < q.[x] -> sgr p.[x] = sp'c.
-Proof. by rewrite -sgr_cp0 sgp => /eqP->; rewrite mulr1. Qed.
-
-Fact hsgpN x : q.[x] < 0 -> sgr p.[x] = - sp'c.
-Proof. by rewrite -sgr_cp0 sgp => /eqP->; rewrite mulrN1. Qed.
-
-Lemma ders0r : p.[a] = 0 -> forall x, x \in `]a, b] -> sgr p.[x] = sp'c.
-Proof.
-move=> pa0 x hx; rewrite hsgp // (@derp0r _ a b) //; first exact: derq_pos.
-by rewrite hornerZ pa0 mulr0.
-Qed.
-
-Lemma derspr : sgr p.[a] = sp'c -> forall x, x \in `[a, b] -> sgr p.[x] = sp'c.
-Proof.
-move=> spa x hx; rewrite hsgp // (@derppr _ a b) //; first exact: derq_pos.
-by rewrite -sgr_cp0 hornerZ sgrM sgr_id spa -expr2 sqr_sg derp_neq0.
-Qed.
-
-Lemma ders0l : p.[b] = 0 -> forall x, x \in `[a, b[ -> sgr p.[x] = -sp'c.
-Proof.
-move=> pa0 x hx; rewrite hsgpN // (@derp0l _ a b) //; first exact: derq_pos.
-by rewrite hornerZ pa0 mulr0.
-Qed.
-
-Lemma derspl :
- sgr p.[b] = -sp'c -> forall x, x \in `[a, b] -> sgr p.[x] = -sp'c.
-Proof.
-move=> spb x hx; rewrite hsgpN // (@derpnl _ a b) //; first exact: derq_pos.
-by rewrite -sgr_cp0 hornerZ sgr_smul spb mulrN -expr2 sqr_sg derp_neq0.
-Qed.
-
-End NoRoot_sg.
-
-Variable (p : {poly R}).
-
-Variables (a b : R).
-
-Section der_root.
-
-Hypothesis der_pos : forall x, x \in `]a, b[ -> (p^`()).[x] > 0.
-
-Lemma derp_root : a <= b -> 0 \in `]p.[a], p.[b][ ->
- { r : R |
- [/\ forall x, x \in `[a, r[ -> p.[x] < 0,
- p.[r] = 0,
- r \in `]a, b[ &
- forall x, x \in `]r, b] -> p.[x] > 0] }.
-Proof.
-move=> leab hpab.
-have /eqP hs : sgr p.[a] * sgr p.[b] == -1.
- by rewrite -sgrM sgr_cp0 pmulr_llt0 ?(itvP hpab).
-case: (ivt_sign leab hs) => r arb pr0; exists r; split => //; last 2 first.
-- by move/eqP: pr0.
-- move=> x rxb; have hd : forall t, t \in `]r, b[ -> 0 < (p^`()).[t].
- by move=> t ht; rewrite der_pos // ?(subitvPl _ ht) //= ?(itvP arb).
- by rewrite (derp0r hd) ?(eqP pr0).
-- move=> x rxb; have hd : forall t, t \in `]a, r[ -> 0 < (p^`()).[t].
- by move=> t ht; rewrite der_pos // ?(subitvPr _ ht) //= ?(itvP arb).
- by rewrite (derp0l hd) ?(eqP pr0).
-Qed.
-
-End der_root.
-
-(* Section der_root_sg. *)
-
-(* Hypothesis der_pos : forall x, x \in `]a, b[ -> (p^`()).[x] != 0. *)
-
-(* Lemma derp_root : a <= b -> sgr p.[a] != sgr p.[b] -> *)
-(* { r : R | *)
-(* [/\ forall x, x \in `[a, r[ -> p.[x] < 0, *)
-(* p.[r] = 0, *)
-(* r \in `]a, b[ & *)
-(* forall x, x \in `]r, b] -> p.[x] > 0] }. *)
-(* Proof. *)
-(* move=> leab hpab. *)
-(* have hs : sgr p.[a] * sgr p.[b] == -1. *)
-(* by rewrite -sgrM sgr_cp0 mulr_lt0_gt0 ?(itvP hpab). *)
-(* case: (ivt_sign ivt leab hs) => r arb pr0; exists r; split => //; last 2 first. *)
-(* - by move/eqP: pr0. *)
-(* - move=> x rxb; have hd : forall t, t \in `]r, b[ -> 0 < (p^`()).[t]. *)
-(* by move=> t ht; rewrite der_pos // ?(subitvPl _ ht) //= ?(itvP arb). *)
-(* by rewrite (derp0r hd) ?(eqP pr0). *)
-(* - move=> x rxb; have hd : forall t, t \in `]a, r[ -> 0 < (p^`()).[t]. *)
-(* by move=> t ht; rewrite der_pos // ?(subitvPr _ ht) //= ?(itvP arb). *)
-(* by rewrite (derp0l hd) ?(eqP pr0). *)
-(* Qed. *)
-
-(* End der_root. *)
-
-End MonotonictyAndRoots.
-
-Section RootsOn.
-
-Variable T : predType R.
-
-Definition roots_on (p : {poly R}) (i : T) (s : seq R) :=
- forall x, (x \in i) && (root p x) = (x \in s).
-
-Lemma roots_onP p i s : roots_on p i s -> {in i, root p =1 mem s}.
-Proof. by move=> hp x hx; move: (hp x); rewrite hx. Qed.
-
-Lemma roots_on_in p i s :
- roots_on p i s -> forall x, x \in s -> x \in i.
-Proof. by move=> hp x; rewrite -hp; case/andP. Qed.
-
-Lemma roots_on_root p i s :
- roots_on p i s -> forall x, x \in s -> root p x.
-Proof. by move=> hp x; rewrite -hp; case/andP. Qed.
-
-Lemma root_roots_on p i s :
- roots_on p i s -> forall x, x \in i -> root p x -> x \in s.
-Proof. by move=> hp x; rewrite -hp=> ->. Qed.
-
-Lemma roots_on_opp p i s : roots_on (- p) i s -> roots_on p i s.
-Proof. by move=> hp x; rewrite -hp rootN. Qed.
-
-Lemma roots_on_nil p i : roots_on p i [::] -> {in i, noroot p}.
-Proof. by move=> hp x hx; move: (hp x); rewrite in_nil hx /=; move->. Qed.
-
-Lemma roots_on_same s' p i s : s =i s' -> (roots_on p i s <-> roots_on p i s').
-Proof. by move=> hss'; split=> hs x; rewrite (hss', (I, hss')). Qed.
-
-End RootsOn.
-
-
-(* (* Symmetry of center a *) *)
-(* Definition symr (a x : R) := a - x. *)
-
-(* Lemma symr_inv : forall a, involutive (symr a). *)
-(* Proof. by move=> a y; rewrite /symr opprD addrA subrr opprK add0r. Qed. *)
-
-(* Lemma symr_inj : forall a, injective (symr a). *)
-(* Proof. by move=> a; apply: inv_inj; apply: symr_inv. Qed. *)
-
-(* Lemma ltr_sym : forall a x y, (symr a x < symr a y) = (y < x). *)
-(* Proof. by move=> a x y; rewrite lter_add2r lter_oppr opprK. Qed. *)
-
-(* Lemma symr_add_itv : forall a b x, *)
-(* (a < symr (a + b) x < b) = (a < x < b). *)
-(* Proof. *)
-(* move=> a b x; rewrite andbC. *)
-(* by rewrite lter_subrA lter_add2r -lter_addlA lter_add2l. *)
-(* Qed. *)
-
-Lemma roots_on_comp p a b s :
- roots_on (p \Po (-'X)) `](-b), (-a)[ (map (-%R) s) <-> roots_on p `]a, b[ s.
-Proof.
-split=> /= hs x; rewrite ?root_comp ?hornerE.
- move: (hs (-x)); rewrite mem_map; last exact: (inv_inj (@opprK _)).
- by rewrite root_comp ?hornerE oppr_itv !opprK.
-rewrite -[x]opprK oppr_itv /= mem_map; last exact: (inv_inj (@opprK _)).
-by move: (hs (-x)); rewrite !opprK.
-Qed.
-
-Lemma min_roots_on p a b x s :
- all (> x) s -> roots_on p `]a, b[ (x :: s) ->
- [/\ x \in `]a, b[, roots_on p `]a, x[ [::], root p x & roots_on p `]x, b[ s].
-Proof.
-move=> lxs hxs.
-have hx: x \in `]a, b[ by rewrite (roots_on_in hxs) ?mem_head.
-rewrite hx (roots_on_root hxs) ?mem_head //.
-split=> // y; move: (hxs y); rewrite ?in_nil ?in_cons /=.
- case hy: (y \in `]a, x[)=> //=.
- rewrite (subitvPr _ hy) //= ?(itvP hx) //= => ->.
- rewrite ltr_eqF ?(itvP hy) //=; apply/negP.
- by move/allP: lxs=> lxs /lxs; rewrite ltrNge ?(itvP hy).
-move/allP:lxs=>lxs; case eyx: (y == _)=> /=.
- case/andP=> hy _; rewrite (eqP eyx).
- rewrite boundl_in_itv /=; symmetry.
- by apply/negP; move/lxs; rewrite ltrr.
-case py0: root; rewrite !(andbT, andbF) //.
-case ys: (y \in s); first by move/lxs:ys; rewrite ?inE => ->; case/andP.
-move/negP; move/negP=> nhy; apply: negbTE; apply: contra nhy.
-by apply: subitvPl; rewrite //= ?(itvP hx).
-Qed.
-
-Lemma max_roots_on p a b x s :
- all (< x) s -> roots_on p `]a, b[ (x :: s) ->
- [/\ x \in `]a, b[, roots_on p `]x, b[ [::], root p x & roots_on p `]a, x[ s].
-Proof.
-move/allP=> lsx /roots_on_comp/=/min_roots_on[].
- apply/allP=> y; rewrite -[y]opprK mem_map.
- by move/lsx; rewrite ltr_oppr opprK.
- exact: (inv_inj (@opprK _)).
-rewrite oppr_itv root_comp !hornerE !opprK=> -> rxb -> rax.
-by split=> //; apply/roots_on_comp.
-Qed.
-
-Lemma roots_on_cons p a b r s :
- sorted <%R (r :: s) -> roots_on p `]a, b[ (r :: s) -> roots_on p `]r, b[ s.
-Proof.
-move=> /= hrs hr.
-have:= (order_path_min (@ltr_trans _) hrs)=> allrs.
-by case: (min_roots_on allrs hr).
-Qed.
-(* move=> p a b r s hp hr x; apply/andP/idP. *)
-(* have:= (order_path_min (@ltr_trans _) hp) => /=; case/andP=> ar1 _. *)
-(* case; move/ooitvP=> rxb rpx; move: (hr x); rewrite in_cons rpx andbT. *)
-(* by rewrite rxb andbT (ltr_trans ar1) 1?eq_sym ?ltr_eqF ?rxb. *)
-(* move=> spx. *)
-(* have xrsp: x \in r :: s by rewrite in_cons spx orbT. *)
-(* rewrite (roots_on_root hr) //. *)
-(* rewrite (roots_on_in hr xrsp); move: hp => /=; case/andP=> _. *)
-(* by move/(order_path_min (@ltr_trans _)); move/allP; move/(_ _ spx)->. *)
-(* Qed. *)
-
-Lemma roots_on_rcons : forall p a b r s,
- sorted <%R (rcons s r) -> roots_on p `]a, b[ (rcons s r)
- -> roots_on p `]a, r[ s.
-Proof.
-move=> p a b r s; rewrite -{1}[s]revK -!rev_cons rev_sorted /=.
-move=> hrs hr.
-have := (order_path_min (rev_trans (@ltr_trans _)) hrs)=> allrrs.
-have allrs: (all (< r) s).
- by apply/allP=> x hx; move/allP:allrrs; apply; rewrite mem_rev.
-move/(@roots_on_same _ _ _ _ (r::s)):hr=>hr.
-case: (max_roots_on allrs (hr _))=> //.
-by move=> x; rewrite mem_rcons.
-Qed.
-
-
-(* move=> p a b r s; rewrite -{1}[s]revK -!rev_cons rev_sorted. *)
-(* rewrite [r :: _]lock /=; unlock; move=> hp hr x; apply/andP/idP. *)
-(* have:= (order_path_min (rev_trans (@ltr_trans _)) hp) => /=. *)
-(* case/andP=> ar1 _; case; move/ooitvP=> axr rpx. *)
-(* move: (hr x); rewrite mem_rcons in_cons rpx andbT axr andTb. *)
-(* by rewrite ((rev_trans (@ltr_trans _) ar1)) ?ltr_eqF ?axr. *)
-(* move=> spx. *)
-(* have xrsp: x \in rcons s r by rewrite mem_rcons in_cons spx orbT. *)
-(* rewrite (roots_on_root hr) //. *)
-(* rewrite (roots_on_in hr xrsp); move: hp => /=; case/andP=> _. *)
-(* move/(order_path_min (rev_trans (@ltr_trans _))); move/allP. *)
-(* by move/(_ x)=> -> //; rewrite mem_rev. *)
-(* Qed. *)
-
-Lemma no_roots_on (p : {poly R}) a b :
- {in `]a, b[, noroot p} -> roots_on p `]a, b[ [::].
-Proof.
-move=> hr x; rewrite in_nil; case hx: (x \in _) => //=.
-by apply: negPf; apply: hr hx.
-Qed.
-
-Lemma monotonic_rootN (p : {poly R}) (a b : R) :
- {in `]a, b[, noroot p^`()} ->
- ((roots_on p `]a, b[ [::]) + ({r : R | roots_on p `]a, b[ [:: r]}))%type.
-Proof.
-move=> hp'; case: (ltrP a b); last first => leab.
- by left => x; rewrite in_nil itv_gte.
-wlog {hp'} hp'sg: p / forall x, x \in `]a, b[ -> sgr (p^`()).[x] = 1.
- move=> hwlog. have := (polyrN0_itv hp').
- move: (mid_in_itvoo leab)=> hm /(_ _ _ hm).
- case: (sgrP _.[mid a b])=> hpm.
- - by move=> norm; move: (hp' _ hm); rewrite rootE hpm eqxx.
- - by move/(hwlog p).
- - move=> hp'N; case: (hwlog (-p))=> [x|h|[r hr]].
- * by rewrite derivE hornerN sgrN=> /hp'N->; rewrite opprK.
- * by left; apply: roots_on_opp.
- * by right; exists r; apply: roots_on_opp.
-have hp'pos: forall x, x \in `]a, b[ -> (p^`()).[x] > 0.
- by move=> x; move/hp'sg; move/eqP; rewrite sgr_cp0.
-case: (lerP 0 p.[a]) => ha.
-- left; apply: no_roots_on => x axb.
- by rewrite rootE gtr_eqF // (@derp0r _ a b) // (subitvPr _ axb) /=.
-- case: (lerP p.[b] 0) => hb.
- + left => x; rewrite in_nil; apply: negbTE; case axb: (x \in `]a, b[) => //=.
- by rewrite rootE ltr_eqF // (@derp0l _ a b) // (subitvPl _ axb) /=.
- + case: (derp_root hp'pos (ltrW leab) _).
- by rewrite ?inE; apply/andP.
- move=> r [h1 h2 h3] h4; right.
- exists r => x; rewrite in_cons in_nil (itv_splitU2 h3).
- case exr: (x == r); rewrite ?(andbT, orbT, andbF, orbF) /=.
- by rewrite rootE (eqP exr) h2 eqxx.
- case px0: root; rewrite (andbT, andbF) //.
- move/eqP: px0=> px0; apply/negP; case/orP=> hx.
- by move: (h1 x); rewrite (subitvPl _ hx) //= px0 ltrr; move/implyP.
- by move: (h4 x); rewrite (subitvPr _ hx) //= px0 ltrr; move/implyP.
-Qed.
-
-(* Inductive polN0 : Type := PolN0 : forall p : {poly R}, p != 0 -> polN0. *)
-
-(* Coercion pol_of_polN0 i := let: PolN0 p _ := i in p. *)
-
-(* Canonical Structure polN0_subType := [subType for pol_of_polN0]. *)
-(* Definition polN0_eqMixin := Eval hnf in [eqMixin of polN0 by <:]. *)
-(* Canonical Structure polN0_eqType := *)
-(* Eval hnf in EqType polN0 polN0_eqMixin. *)
-(* Definition polN0_choiceMixin := [choiceMixin of polN0 by <:]. *)
-(* Canonical Structure polN0_choiceType := *)
-(* Eval hnf in ChoiceType polN0 polN0_choiceMixin. *)
-
-(* Todo : Lemmas about operations of intervall :
- itversection, restriction and splitting *)
-Lemma cat_roots_on (p : {poly R}) a b x :
- x \in `]a, b[ -> ~~ (root p x) ->
- forall s s', sorted <%R s -> sorted <%R s' ->
- roots_on p `]a, x[ s -> roots_on p `]x, b[ s' ->
- roots_on p `]a, b[ (s ++ s').
-Proof.
-move=> hx /= npx0 s; elim: s a hx => [|y s ihs] a hx s' //= ss ss'.
- move/roots_on_nil=> hax hs' y.
- rewrite -hs'; case py0: root; rewrite ?(andbT, andbF) //.
- rewrite (itv_splitU2 hx); case: (y \in `]x, b[); rewrite ?orbF ?orbT //=.
- apply/negP; case/orP; first by move/hax; rewrite py0.
- by move/eqP=> exy; rewrite -exy py0 in npx0.
-move/min_roots_on; rewrite order_path_min //; last exact: ltr_trans.
-case=> // hy hay py0 hs hs' z.
-rewrite in_cons; case ezy: (z == y)=> /=.
- by rewrite (eqP ezy) py0 andbT (subitvPr _ hy) //= ?(itvP hx).
-rewrite -(ihs y) //; last exact: path_sorted ss; last first.
- by rewrite inE /= (itvP hx) (itvP hy).
-case pz0: root; rewrite ?(andbT, andbF) //.
-rewrite (@itv_splitU2 _ y); last by rewrite (subitvPr _ hy) //= (itvP hx).
-rewrite ezy /=; case: (z \in `]y, b[); rewrite ?orbF ?orbT //.
-by apply/negP=> hz; move: (hay z); rewrite hz pz0 in_nil.
-Qed.
-
-CoInductive roots_spec (p : {poly R}) (i : pred R) (s : seq R) :
- {poly R} -> bool -> seq R -> Type :=
-| Roots0 of p = 0 :> {poly R} & s = [::] : roots_spec p i s 0 true [::]
-| Roots of p != 0 & roots_on p i s
- & sorted <%R s : roots_spec p i s p false s.
-
-(* still much too long *)
-Lemma itv_roots (p :{poly R}) (a b : R) :
- {s : seq R & roots_spec p (topred `]a, b[) s p (p == 0) s}.
-Proof.
-case p0: (_ == 0).
- by rewrite (eqP p0); exists [::]; constructor.
-elim: (size p) {-2}p (leqnn (size p)) p0 a b => {p} [| n ihn] p sp p0 a b.
- by exists [::]; move: p0; rewrite -size_poly_eq0 -leqn0 sp.
-move/negbT: (p0)=> np0.
-case p'0 : (p^`() == 0).
- move: p'0; rewrite -derivn1 -derivn_poly0; move/size1_polyC => pC.
- exists [::]; constructor=> // x; rewrite in_nil pC rootC; apply: negPf.
- by rewrite negb_and -polyC_eq0 -pC p0 orbT.
-move/negbT: (p'0) => np'0.
-have sizep' : (size p^`() <= n)%N.
- rewrite -ltnS; apply: leq_trans sp; rewrite size_deriv prednK // lt0n.
- by rewrite size_poly_eq0 p0.
-case: (ihn _ sizep' p'0 a b) => sp' ih {ihn}.
-case ltab : (a < b); last first.
- exists [::]; constructor=> // x; rewrite in_nil.
- case axb : (x \in _) => //=.
- by case/andP: axb => ax xb; move: ltab; rewrite (ltr_trans ax xb).
-elim: sp' a b ltab ih => [|r1 sp' hsp'] a b ltab hp'.
- case: hp' np'0; rewrite ?eqxx // => np'0 hroots' _ _.
- move/roots_on_nil : hroots' => hroots'.
- case: (monotonic_rootN hroots') => [h| [r rh]].
- by exists [::]; constructor.
- by exists [:: r]; constructor=> //=; rewrite andbT.
-case: hp' np'0; rewrite ?eqxx // => np'0 hroots' /= hpath' _.
-case: (min_roots_on _ hroots').
- by rewrite order_path_min //; apply: ltr_trans.
-move=> hr1 har1 p'r10 hr1b.
-case: (hsp' r1 b); first by rewrite (itvP hr1).
- by constructor=> //; rewrite (path_sorted hpath').
-move=> s spec_s.
-case: spec_s np0; rewrite ?eqxx //.
-move=> np0 hroot hsort _.
-move: (roots_on_nil har1).
-case pr1 : (root p r1); case/monotonic_rootN => hrootsl; last 2 first.
-- exists s; constructor=> //.
- by rewrite -[s]cat0s; apply: (cat_roots_on hr1)=> //; rewrite pr1.
-- case: hrootsl=> r hr; exists (r::s); constructor=> //=.
- by rewrite -cat1s; apply: (cat_roots_on hr1)=> //; rewrite pr1.
- rewrite path_min_sorted // => y; rewrite -hroot; case/andP=> hy _.
- rewrite (@ltr_trans _ r1) ?(itvP hy) //.
- by rewrite (itvP (roots_on_in hr (mem_head _ _))).
-- exists (r1::s); constructor=> //=; last first.
- rewrite path_min_sorted=> // y; rewrite -hroot.
- by case/andP; move/itvP->.
- move=> x; rewrite in_cons; case exr1: (x == r1)=> /=.
- by rewrite (eqP exr1) pr1 andbT.
- rewrite -hroot; case px: root; rewrite ?(andbT, andbF) //.
- rewrite (itv_splitU2 hr1) exr1 /=.
- case: (_ \in `]r1, _[); rewrite ?(orbT, orbF) //.
- by apply/negP=> hx; move: (hrootsl x); rewrite hx px in_nil.
-- case: hrootsl => r0 hrootsl.
- move/min_roots_on:hrootsl; case=> // hr0 har0 pr0 hr0r1.
- exists [:: r0, r1 & s]; constructor=> //=; last first.
- rewrite (itvP hr0) /= path_min_sorted // => y.
- by rewrite -hroot; case/andP; move/itvP->.
- move=> y; rewrite !in_cons (itv_splitU2 hr1) (itv_splitU2 hr0).
- case eyr0: (y == r0); rewrite ?(orbT, orbF, orTb, orFb).
- by rewrite (eqP eyr0) pr0.
- case eyr1: (y == r1); rewrite ?(orbT, orbF, orTb, orFb).
- by rewrite (eqP eyr1) pr1.
- rewrite -hroot; case py0: root; rewrite ?(andbF, andbT) //.
- case: (_ \in `]r1, _[); rewrite ?(orbT, orbF) //.
- apply/negP; case/orP=> hy; first by move: (har0 y); rewrite hy py0 in_nil.
- by move: (hr0r1 y); rewrite hy py0 in_nil.
-Qed.
-
-Definition roots (p : {poly R}) a b := projT1 (itv_roots p a b).
-
-Lemma rootsP p a b :
- roots_spec p (topred `]a, b[) (roots p a b) p (p == 0) (roots p a b).
-Proof. by rewrite /roots; case hp: itv_roots. Qed.
-
-Lemma roots0 a b : roots 0 a b = [::].
-Proof. by case: rootsP=> //=; rewrite eqxx. Qed.
-
-Lemma roots_on_roots : forall p a b, p != 0 ->
- roots_on p `]a, b[ (roots p a b).
-Proof. by move=> a b p; case: rootsP. Qed.
-Hint Resolve roots_on_roots.
-
-Lemma sorted_roots a b p : sorted <%R (roots p a b).
-Proof. by case: rootsP. Qed.
-Hint Resolve sorted_roots.
-
-Lemma path_roots p a b : path <%R a (roots p a b).
-Proof.
-case: rootsP=> //= p0 hp sp; rewrite path_min_sorted //.
-by move=> y; rewrite -hp; case/andP; move/itvP->.
-Qed.
-Hint Resolve path_roots.
-
-Lemma root_is_roots (p : {poly R}) (a b : R) :
- p != 0 -> forall x, x \in `]a, b[ -> root p x = (x \in roots p a b).
-Proof. by case: rootsP=> // p0 hs ps _ y hy /=; rewrite -hs hy. Qed.
-
-Lemma root_in_roots (p : {poly R}) a b :
- p != 0 -> forall x, x \in `]a, b[ -> root p x -> x \in (roots p a b).
-Proof. by move=> p0 x axb rpx; rewrite -root_is_roots. Qed.
-
-Lemma root_roots p a b x : x \in roots p a b -> root p x.
-Proof. by case: rootsP=> // p0 <- _; case/andP. Qed.
-
-Lemma roots_nil p a b : p != 0 ->
- roots p a b = [::] -> {in `]a, b[, noroot p}.
-Proof.
-case: rootsP => // p0 hs ps _ s0 x axb.
-by move: (hs x); rewrite s0 in_nil !axb /= => ->.
-Qed.
-
-Lemma roots_in p a b x : x \in roots p a b -> x \in `]a, b[.
-Proof. by case: rootsP=> //= np0 ron_p *; apply: (roots_on_in ron_p). Qed.
-
-Lemma rootsEba p a b : b <= a -> roots p a b = [::].
-Proof.
-case: rootsP=> // p0; case: (roots _ _ _) => [|x s] hs ps ba //;
-by move: (hs x); rewrite itv_gte //= mem_head.
-Qed.
-
-Lemma roots_on_uniq p a b s1 s2 :
- sorted <%R s1 -> sorted <%R s2 ->
- roots_on p `]a, b[ s1 -> roots_on p `]a, b[ s2 -> s1 = s2.
-Proof.
-elim: s1 p a b s2 => [| r1 s1 ih] p a b [| r2 s2] ps1 ps2 rs1 rs2 //.
-- have rpr2 : root p r2 by apply: (roots_on_root rs2); rewrite mem_head.
- have abr2 : r2 \in `]a, b[ by apply: (roots_on_in rs2); rewrite mem_head.
- by have:= (rs1 r2); rewrite rpr2 !abr2 in_nil.
-- have rpr1 : root p r1 by apply: (roots_on_root rs1); rewrite mem_head.
- have abr1 : r1 \in `]a, b[ by apply: (roots_on_in rs1); rewrite mem_head.
- by have:= (rs2 r1); rewrite rpr1 !abr1 in_nil.
-- have er1r2 : r1 = r2.
- move: (rs1 r2); rewrite (roots_on_root rs2) ?mem_head //.
- rewrite !(roots_on_in rs2) ?mem_head //= in_cons.
- move/(@sym_eq _ true); case/orP => hr2; first by rewrite (eqP hr2).
- move: ps1=> /=; move/(order_path_min (@ltr_trans R)); move/allP.
- move/(_ r2 hr2) => h1.
- move: (rs2 r1); rewrite (roots_on_root rs1) ?mem_head //.
- rewrite !(roots_on_in rs1) ?mem_head //= in_cons.
- move/(@sym_eq _ true); case/orP => hr1; first by rewrite (eqP hr1).
- move: ps2=> /=; move/(order_path_min (@ltr_trans R)); move/allP.
- by move/(_ r1 hr1); rewrite ltrNge ltrW.
-congr (_ :: _) => //; rewrite er1r2 in ps1 rs1.
-have h3 := (roots_on_cons ps1 rs1).
-have h4 := (roots_on_cons ps2 rs2).
-move: ps1 ps2=> /=; move/path_sorted=> hs1; move/path_sorted=> hs2.
-exact: (ih p _ b _ hs1 hs2 h3 h4).
-Qed.
-
-Lemma roots_eq (p q : {poly R}) (a b : R) :
- p != 0 -> q != 0 ->
- ({in `]a, b[, root p =1 root q} <-> roots p a b = roots q a b).
-Proof.
-move=> p0 q0.
-case hab : (a < b); last first.
- split; first by rewrite !rootsEba // lerNgt hab.
- move=> _ x. rewrite !inE; case/andP=> ax xb.
- by move: hab; rewrite (@ltr_trans _ x) /=.
-split=> hpq.
- apply: (@roots_on_uniq p a b); rewrite ?path_roots ?p0 ?q0 //.
- by apply: roots_on_roots.
- rewrite /roots_on => x; case hx: (_ \in _).
- by rewrite -hx hpq //; apply: roots_on_roots.
- by rewrite /= -(andFb (q.[x] == 0)) -hx; apply: roots_on_roots.
-move=> x axb /=.
-by rewrite (@root_is_roots q a b) // (@root_is_roots p a b) // hpq.
-Qed.
-
-Lemma roots_opp p : roots (- p) =2 roots p.
-Proof.
-move=> a b; case p0 : (p == 0); first by rewrite (eqP p0) oppr0.
-by apply/roots_eq=> [||x]; rewrite ?oppr_eq0 ?p0 ?rootN.
-Qed.
-
-Lemma no_root_roots (p : {poly R}) a b :
- {in `]a, b[ , noroot p} -> roots p a b = [::].
-Proof.
-move=> hr; case: rootsP => // p0 hs ps.
-apply: (@roots_on_uniq p a b)=> // x; rewrite in_nil.
-by apply/negP; case/andP; move/hr; move/negPf->.
-Qed.
-
-Lemma head_roots_on_ge p a b s :
- a < b -> roots_on p `]a, b[ s -> a < head b s.
-Proof.
-case: s => [|x s] ab // /(_ x).
-by rewrite in_cons eqxx; case/andP; case/andP.
-Qed.
-
-Lemma head_roots_ge : forall p a b, a < b -> a < head b (roots p a b).
-Proof.
-by move=> p a b; case: rootsP=> // *; apply: head_roots_on_ge.
-Qed.
-
-Lemma last_roots_on_le p a b s :
- a < b -> roots_on p `]a, b[ s -> last a s < b.
-Proof.
-case: s => [|x s] ab rs //.
-by rewrite (itvP (roots_on_in rs _)) //= mem_last.
-Qed.
-
-Lemma last_roots_le p a b : a < b -> last a (roots p a b) < b.
-Proof. by case: rootsP=> // *; apply: last_roots_on_le. Qed.
-
-Lemma roots_uniq p a b s :
- p != 0 -> roots_on p `]a, b[ s -> sorted <%R s -> s = roots p a b.
-Proof.
-case: rootsP=> // p0 hs' ps' _ hs ss.
-exact: (@roots_on_uniq p a b)=> //.
-Qed.
-
-Lemma roots_cons p a b x s :
- (roots p a b == x :: s)
- = [&& p != 0, x \in `]a, b[,
- roots p a x == [::], root p x & roots p x b == s].
-Proof.
-case: rootsP=> // p0 hs' ps' /=.
-apply/idP/idP.
- move/eqP=> es'; move: ps' hs'; rewrite es' /= => sxs.
- case/min_roots_on; first by apply: order_path_min=> //; apply: ltr_trans.
- move=> -> rax px0 rxb.
- rewrite px0 (@roots_uniq p a x [::]) // (@roots_uniq p x b s) ?eqxx //=.
- by move/path_sorted:sxs.
-case: rootsP p0=> // p0 rax sax _.
-case/and3P=> hx hax; rewrite (eqP hax) in rax sax.
-case: rootsP p0=> // p0 rxb sxb _.
-case/andP=> px0 hxb; rewrite (eqP hxb) in rxb sxb.
-rewrite [_ :: _](@roots_uniq p a b) //; last first.
- rewrite /= path_min_sorted // => y.
- by rewrite -(eqP hxb); move/roots_in; move/itvP->.
-move=> y; rewrite (itv_splitU2 hx) !andb_orl in_cons.
-case hy: (y == x); first by rewrite (eqP hy) px0 orbT.
-by rewrite andFb orFb rax rxb in_nil.
-Qed.
-
-Lemma roots_rcons p a b x s :
- (roots p a b == rcons s x) =
- [&& p != 0, x \in `]a , b[,
- roots p x b == [::], root p x & roots p a x == s].
-Proof.
-case: rootsP; first by case: s.
-move=> // p0 hs' ps' /=.
-apply/idP/idP.
- move/eqP=> es'; move: ps' hs'; rewrite es' /= => sxs.
- have hsx: rcons s x =i x :: rev s.
- by move=> y; rewrite mem_rcons !in_cons mem_rev.
- move/(roots_on_same _ _ hsx).
- case/max_roots_on.
- move: sxs; rewrite -[rcons _ _]revK rev_sorted rev_rcons.
- by apply: order_path_min=> u v w /=; move/(ltr_trans _); apply.
- move=> -> rax px0 rxb.
- move/(@roots_on_same _ s): rxb; move/(_ (mem_rev _))=> rxb.
- rewrite px0 (@roots_uniq p x b [::]) // (@roots_uniq p a x s) ?eqxx //=.
- move: sxs; rewrite -[rcons _ _]revK rev_sorted rev_rcons.
- by move/path_sorted; rewrite -rev_sorted revK.
-case: rootsP p0=> // p0 rax sax _.
-case/and3P=> hx hax; rewrite (eqP hax) in rax sax.
-case: rootsP p0=> // p0 rxb sxb _.
-case/andP=> px0 hxb; rewrite (eqP hxb) in rxb sxb.
-rewrite [rcons _ _](@roots_uniq p a b) //; last first.
- rewrite -[rcons _ _]revK rev_sorted rev_rcons /= path_min_sorted.
- by rewrite -rev_sorted revK.
- move=> y; rewrite mem_rev; rewrite -(eqP hxb).
- by move/roots_in; move/itvP->.
-move=> y; rewrite (itv_splitU2 hx) mem_rcons in_cons !andb_orl.
-case hy: (y == x); first by rewrite (eqP hy) px0 orbT.
-by rewrite rxb rax in_nil !orbF.
-Qed.
-
-Section NeighborHood.
-
-Implicit Types a b : R.
-
-Implicit Types p : {poly R}.
-
-Definition next_root (p : {poly R}) x b :=
- if p == 0 then x else head (maxr b x) (roots p x b).
-
-Lemma next_root0 a b : next_root 0 a b = a.
-Proof. by rewrite /next_root eqxx. Qed.
-
-CoInductive next_root_spec (p : {poly R}) x b : bool -> R -> Type :=
-| NextRootSpec0 of p = 0 : next_root_spec p x b true x
-| NextRootSpecRoot y of p != 0 & p.[y] = 0 & y \in `]x, b[
- & {in `]x, y[, forall z, ~~(root p z)} : next_root_spec p x b true y
-| NextRootSpecNoRoot c of p != 0 & c = maxr b x
- & {in `]x, b[, forall z, ~~(root p z)} : next_root_spec p x b (p.[c] == 0) c.
-
-Lemma next_rootP (p : {poly R}) a b :
- next_root_spec p a b (p.[next_root p a b] == 0) (next_root p a b).
-Proof.
-rewrite /next_root /=; case hs: roots => [|x s] /=.
- case: (altP (p =P 0))=> p0.
- by rewrite {2}p0 hornerC eqxx; constructor; rewrite p0.
- by constructor=> // y hy; apply: (roots_nil p0 hs).
-move/eqP: hs; rewrite roots_cons; case/and5P=> p0 hx; move/eqP=> rap rpx rx.
-rewrite (negPf p0) (rootPt rpx); constructor=> //; first by move/eqP: rpx.
-by move=> y hy /=; move/(roots_nil p0): (rap); apply.
-Qed.
-
-Lemma next_root_in p a b : next_root p a b \in `[a, maxr b a].
-Proof.
-case: next_rootP => [p0|y np0 py0 hy _|c np0 hc _].
-* by rewrite bound_in_itv /= ler_maxr lerr orbT.
-* by apply: subitvP hy=> /=; rewrite ler_maxr !lerr.
-* by rewrite hc bound_in_itv /= ler_maxr lerr orbT.
-Qed.
-
-Lemma next_root_gt p a b : a < b -> p != 0 -> next_root p a b > a.
-Proof.
-move=> ab np0; case: next_rootP=> [p0|y _ py0 hy _|c _ -> _].
-* by rewrite p0 eqxx in np0.
-* by rewrite (itvP hy).
-* by rewrite maxr_l // ltrW.
-Qed.
-
-Lemma next_noroot p a b : {in `]a, (next_root p a b)[, noroot p}.
-Proof.
-move=> z; case: next_rootP; first by rewrite itv_xx.
- by move=> y np0 py0 hy hp hz; rewrite (negPf (hp _ _)).
-move=> c p0 -> hp hz; rewrite (negPf (hp _ _)) //.
-by case: maxrP hz; last by rewrite itv_xx.
-Qed.
-
-Lemma is_next_root p a b x :
- next_root_spec p a b (root p x) x -> x = next_root p a b.
-Proof.
-case; first by move->; rewrite /next_root eqxx.
- move=> y; case: next_rootP; first by move->; rewrite eqxx.
- move=> y' np0 py'0 hy' hp' _ py0 hy hp.
- wlog: y y' hy' hy hp' hp py0 py'0 / y <= y'.
- by case/orP: (ler_total y y')=> lyy' hw; [|symmetry]; apply: hw.
- case: ltrgtP=> // hyy' _; move: (hp' y).
- by rewrite rootE py0 eqxx inE /= (itvP hy) hyy'; move/(_ isT).
- move=> c p0 ->; case: maxrP=> hab; last by rewrite itv_gte //= ltrW.
- by move=> hpz _ py0 hy; move/hpz:hy; rewrite rootE py0 eqxx.
-case: next_rootP => //; first by move->; rewrite eqxx.
- by move=> y np0 py0 hy _ c _ _; move/(_ _ hy); rewrite rootE py0 eqxx.
-by move=> c _ -> _ c' _ ->.
-Qed.
-
-Definition prev_root (p : {poly R}) a x :=
- if p == 0 then x else last (minr a x) (roots p a x).
-
-Lemma prev_root0 a b : prev_root 0 a b = b.
-Proof. by rewrite /prev_root eqxx. Qed.
-
-CoInductive prev_root_spec (p : {poly R}) a x : bool -> R -> Type :=
-| PrevRootSpec0 of p = 0 : prev_root_spec p a x true x
-| PrevRootSpecRoot y of p != 0 & p.[y] = 0 & y \in`]a, x[
- & {in `]y, x[, forall z, ~~(root p z)} : prev_root_spec p a x true y
-| PrevRootSpecNoRoot c of p != 0 & c = minr a x
- & {in `]a, x[, forall z, ~~(root p z)} : prev_root_spec p a x (p.[c] == 0) c.
-
-Lemma prev_rootP (p : {poly R}) a b :
- prev_root_spec p a b (p.[prev_root p a b] == 0) (prev_root p a b).
-Proof.
-rewrite /prev_root /=; move hs: (roots _ _ _)=> s.
-case: (lastP s) hs=> {s} [|s x] hs /=.
- case: (altP (p =P 0))=> p0.
- by rewrite {2}p0 hornerC eqxx; constructor; rewrite p0.
- by constructor=> // y hy; apply: (roots_nil p0 hs).
-rewrite last_rcons; move/eqP: hs.
-rewrite roots_rcons; case/and5P=> p0 hx; move/eqP=> rap rpx rx.
-rewrite (negPf p0) (rootPt rpx); constructor=> //; first by move/eqP: rpx.
-by move=> y hy /=; move/(roots_nil p0): (rap); apply.
-Qed.
-
-Lemma prev_root_in p a b : prev_root p a b \in `[minr a b, b].
-Proof.
-case: prev_rootP => [p0|y np0 py0 hy _|c np0 hc _].
-* by rewrite bound_in_itv /= ler_minl lerr orbT.
-* by apply: subitvP hy=> /=; rewrite ler_minl !lerr.
-* by rewrite hc bound_in_itv /= ler_minl lerr orbT.
-Qed.
-
-Lemma prev_noroot p a b : {in `](prev_root p a b), b[, noroot p}.
-Proof.
-move=> z; case: prev_rootP; first by rewrite itv_xx.
- by move=> y np0 py0 hy hp hz; rewrite (negPf (hp _ _)).
-move=> c np0 ->; case: minrP=> hab; last by rewrite itv_xx.
-by move=> hp hz; rewrite (negPf (hp _ _)).
-Qed.
-
-Lemma prev_root_lt p a b : a < b -> p != 0 -> prev_root p a b < b.
-Proof.
-move=> ab np0; case: prev_rootP=> [p0|y _ py0 hy _|c _ -> _].
-* by rewrite p0 eqxx in np0.
-* by rewrite (itvP hy).
-* by rewrite minr_l // ltrW.
-Qed.
-
-Lemma is_prev_root p a b x :
- prev_root_spec p a b (root p x) x -> x = prev_root p a b.
-Proof.
-case; first by move->; rewrite /prev_root eqxx.
- move=> y; case: prev_rootP; first by move->; rewrite eqxx.
- move=> y' np0 py'0 hy' hp' _ py0 hy hp.
- wlog: y y' hy' hy hp' hp py0 py'0 / y <= y'.
- by case/orP: (ler_total y y')=> lyy' hw; [|symmetry]; apply: hw.
- case: ltrgtP=> // hyy' _; move/implyP: (hp y').
- by rewrite rootE py'0 eqxx inE /= (itvP hy') hyy'.
- by move=> c _ _ hpz _ py0 hy; move/hpz:hy; rewrite rootE py0 eqxx.
-case: prev_rootP=> //; first by move->; rewrite eqxx.
- move=> y ? py0 hy _ c _ ->; case: minrP hy=> hab; last first.
- by rewrite itv_gte //= ltrW.
- by move=> hy; move/(_ _ hy); rewrite rootE py0 eqxx.
-by move=> c _ -> _ c' _ ->.
-Qed.
-
-Definition neighpr p a b := `]a, (next_root p a b)[.
-
-Definition neighpl p a b := `](prev_root p a b), b[.
-
-Lemma neighpl_root p a x : {in neighpl p a x, noroot p}.
-Proof. exact: prev_noroot. Qed.
-
-Lemma sgr_neighplN p a x :
- ~~ root p x -> {in neighpl p a x, forall y, (sgr p.[y] = sgr p.[x])}.
-Proof.
-rewrite /neighpl=> nrpx /= y hy.
-apply: (@polyrN0_itv `[y, x]); do ?by rewrite bound_in_itv /= (itvP hy).
-move=> z; rewrite (@itv_splitU _ x false) ?itv_xx /=; last first.
-(* Todo : Lemma itv_splitP *)
- by rewrite bound_in_itv /= (itvP hy).
-rewrite orbC => /predU1P[-> // | hz].
-rewrite (@prev_noroot _ a x) //.
-by apply: subitvPl hz; rewrite /= (itvP hy).
-Qed.
-
-Lemma sgr_neighpl_same p a x :
- {in neighpl p a x &, forall y z, (sgr p.[y] = sgr p.[z])}.
-Proof.
-by rewrite /neighpl=> y z *; apply: (polyrN0_itv (@prev_noroot p a x)).
-Qed.
-
-Lemma neighpr_root p x b : {in neighpr p x b, noroot p}.
-Proof. exact: next_noroot. Qed.
-
-Lemma sgr_neighprN p x b :
- p.[x] != 0 -> {in neighpr p x b, forall y, (sgr p.[y] = sgr p.[x])}.
-Proof.
-rewrite /neighpr=> nrpx /= y hy; symmetry.
-apply: (@polyrN0_itv `[x, y]); do ?by rewrite bound_in_itv /= (itvP hy).
-move=> z; rewrite (@itv_splitU _ x true) ?itv_xx /=; last first.
-(* Todo : Lemma itv_splitP *)
- by rewrite bound_in_itv /= (itvP hy).
-case/orP=> [|hz]; first by rewrite inE /=; move/eqP->.
-rewrite (@next_noroot _ x b) //.
-by apply: subitvPr hz; rewrite /= (itvP hy).
-Qed.
-
-Lemma sgr_neighpr_same p x b :
- {in neighpr p x b &, forall y z, (sgr p.[y] = sgr p.[z])}.
-Proof.
-by rewrite /neighpl=> y z *; apply: (polyrN0_itv (@next_noroot p x b)).
-Qed.
-
-Lemma uniq_roots a b p : uniq (roots p a b).
-Proof.
-case p0: (p == 0); first by rewrite (eqP p0) roots0.
-by apply: (@sorted_uniq _ <%R); [apply: ltr_trans | apply: ltrr|].
-Qed.
-
-Hint Resolve uniq_roots.
-
-Lemma in_roots p (a b x : R) :
- (x \in roots p a b) = [&& root p x, x \in `]a, b[ & p != 0].
-Proof.
-case: rootsP=> //=; first by rewrite in_nil !andbF.
-by move=> p0 hr sr; rewrite andbT -hr andbC.
-Qed.
-
-(* Todo : move to polyorder => need char 0 *)
-Lemma gdcop_eq0 p q : (gdcop p q == 0) = (q == 0) && (p != 0).
-Proof.
-case: (eqVneq q 0) => [-> | q0].
- rewrite gdcop0 /= eqxx /=.
- by case: (eqVneq p 0) => [-> | pn0]; rewrite ?(negPf pn0) eqxx ?oner_eq0.
-rewrite /gdcop; move: {-1}(size q) (leqnn (size q))=> k hk.
-case: (eqVneq p 0) => [-> | p0].
- rewrite eqxx andbF; apply: negPf.
- elim: k q q0 {hk} => [|k ihk] q q0 /=; first by rewrite eqxx oner_eq0.
- case: ifP => _ //.
- by apply: ihk; rewrite gcdp0 divpp ?q0 // polyC_eq0; apply/lc_expn_scalp_neq0.
-rewrite p0 (negPf q0) /=; apply: negPf.
-elim: k q q0 hk => [|k ihk] /= q q0 hk.
- by move: hk q0; rewrite leqn0 size_poly_eq0; move->.
-case: ifP=> cpq; first by rewrite (negPf q0).
-apply: ihk.
- rewrite divpN0; last by rewrite gcdp_eq0 negb_and q0.
- by rewrite dvdp_leq // dvdp_gcdl.
-rewrite -ltnS; apply: leq_trans hk; move: (dvdp_gcdl q p); rewrite dvdp_eq.
-move/eqP=> eqq; move/(f_equal (fun x : {poly R} => size x)): (eqq).
-rewrite size_scale; last exact: lc_expn_scalp_neq0.
-have gcdn0 : gcdp q p != 0 by rewrite gcdp_eq0 negb_and q0.
-have qqn0 : q %/ gcdp q p != 0.
- apply: contraTneq q0; rewrite negbK => e.
- move: (scaler_eq0 (lead_coef (gcdp q p) ^+ scalp q (gcdp q p)) q).
- by rewrite (negPf (lc_expn_scalp_neq0 _ _)) /=; move<-; rewrite eqq e mul0r.
-move->; rewrite size_mul //; case sgcd: (size (gcdp q p)) => [|n].
- by move/eqP: sgcd gcdn0; rewrite size_poly_eq0; move->.
-case: n sgcd => [|n]; first by move/eqP; rewrite size_poly_eq1 gcdp_eqp1 cpq.
-by rewrite addnS /= -{1}[size (_ %/ _)]addn0 ltn_add2l.
-Qed.
-
-Lemma roots_mul a b : a < b -> forall p q,
- p != 0 -> q != 0 -> perm_eq (roots (p*q) a b)
- (roots p a b ++ roots ((gdcop p q)) a b).
-Proof.
-move=> hab p q np0 nq0.
-apply: uniq_perm_eq; first exact: uniq_roots.
- rewrite cat_uniq ?uniq_roots andbT /=; apply/hasPn=> x /=.
- move/root_roots; rewrite root_gdco //; case/andP=> _.
- by rewrite in_roots !negb_and=> ->.
-move=> x; rewrite mem_cat !in_roots root_gdco //.
-rewrite rootM mulf_eq0 gdcop_eq0 negb_and.
-case: (x \in `]_, _[); last by rewrite !andbF.
-by rewrite negb_or !np0 !nq0 !andbT /=; do 2?case: root=> //=.
-Qed.
-
-Lemma roots_mul_coprime a b :
- a < b ->
- forall p q, p != 0 -> q != 0 -> coprimep p q ->
- perm_eq (roots (p * q) a b)
- (roots p a b ++ roots q a b).
-Proof.
-move=> hab p q np0 nq0 cpq.
-rewrite (perm_eq_trans (roots_mul hab np0 nq0)) //.
-suff ->: roots (gdcop p q) a b = roots q a b by apply: perm_eq_refl.
-case: gdcopP=> r rq hrp; move/(_ q (dvdpp _)).
-rewrite coprimep_sym; move/(_ cpq)=> qr.
-have erq : r %= q by rewrite /eqp rq qr.
-(* Todo : relate eqp with roots *)
-apply/roots_eq=> // [|x hx]; last exact: eqp_root.
-by rewrite -size_poly_eq0 (eqp_size erq) size_poly_eq0.
-Qed.
-
-
-Lemma next_rootM a b (p q : {poly R}) :
- next_root (p * q) a b = minr (next_root p a b) (next_root q a b).
-Proof.
-symmetry; apply: is_next_root.
-wlog: p q / next_root p a b <= next_root q a b.
- case: minrP=> hpq; first by move/(_ _ _ hpq); case: minrP hpq.
- by move/(_ _ _ (ltrW hpq)); rewrite mulrC minrC; case: minrP hpq.
-case: minrP=> //; case: next_rootP=> [|y np0 py0 hy|c np0 ->] hp hpq _.
-* by rewrite hp mul0r root0; constructor.
-* rewrite rootM; move/rootP:(py0)->; constructor=> //.
- - by rewrite mulf_neq0 //; case: next_rootP hpq; rewrite // (itvP hy).
- - by rewrite hornerM py0 mul0r.
- - move=> z hz /=; rewrite rootM negb_or ?hp //.
- by rewrite (@next_noroot _ a b) //; apply: subitvPr hz.
-* case: (altP (q =P 0))=> q0.
- move: hpq; rewrite q0 mulr0 root0 next_root0 ler_maxl lerr andbT.
- by move=> hba; rewrite maxr_r //; constructor.
- constructor=> //; first by rewrite mulf_neq0.
- move=> z hz /=; rewrite rootM negb_or ?hp //.
- rewrite (@next_noroot _ a b) //; apply: subitvPr hz=> /=.
- by move: hpq; rewrite ler_maxl; case/andP.
-Qed.
-
-Lemma neighpr_mul a b p q :
- (neighpr (p * q) a b) =i [predI (neighpr p a b) & (neighpr q a b)].
-Proof.
-move=> x; rewrite inE /= !inE /= next_rootM.
-by case: (a < x); rewrite // ltr_minr.
-Qed.
-
-Lemma prev_rootM a b (p q : {poly R}) :
- prev_root (p * q) a b = maxr (prev_root p a b) (prev_root q a b).
-Proof.
-symmetry; apply: is_prev_root.
-wlog: p q / prev_root p a b >= prev_root q a b.
- case: maxrP=> hpq; first by move/(_ _ _ hpq); case: maxrP hpq.
- by move/(_ _ _ (ltrW hpq)); rewrite mulrC maxrC; case: maxrP hpq.
-case: maxrP=> //; case: (@prev_rootP p)=> [|y np0 py0 hy|c np0 ->] hp hpq _.
-* by rewrite hp mul0r root0; constructor.
-* rewrite rootM; move/rootP:(py0)->; constructor=> //.
- - by rewrite mulf_neq0 //; case: prev_rootP hpq; rewrite // (itvP hy).
- - by rewrite hornerM py0 mul0r.
- - move=> z hz /=; rewrite rootM negb_or ?hp //.
- by rewrite (@prev_noroot _ a b) //; apply: subitvPl hz.
-* case: (altP (q =P 0))=> q0.
- move: hpq; rewrite q0 mulr0 root0 prev_root0 ler_minr lerr andbT.
- by move=> hba; rewrite minr_r //; constructor.
- constructor=> //; first by rewrite mulf_neq0.
- move=> z hz /=; rewrite rootM negb_or ?hp //.
- rewrite (@prev_noroot _ a b) //; apply: subitvPl hz=> /=.
- by move: hpq; rewrite ler_minr; case/andP.
-Qed.
-
-Lemma neighpl_mul a b p q :
- (neighpl (p * q) a b) =i [predI (neighpl p a b) & (neighpl q a b)].
-Proof.
-move=> x; rewrite !inE /= prev_rootM.
-by case: (x < b); rewrite // ltr_maxl !(andbT, andbF).
-Qed.
-
-Lemma neighpr_wit p x b : x < b -> p != 0 -> {y | y \in neighpr p x b}.
-Proof.
-move=> xb; exists (mid x (next_root p x b)).
-by rewrite mid_in_itv //= next_root_gt.
-Qed.
-
-Lemma neighpl_wit p a x : a < x -> p != 0 -> {y | y \in neighpl p a x}.
-Proof.
-move=> xb; exists (mid (prev_root p a x) x).
-by rewrite mid_in_itv //= prev_root_lt.
-Qed.
-
-End NeighborHood.
-
-Section SignRight.
-
-Definition sgp_right (p : {poly R}) x :=
- let fix aux (p : {poly R}) n :=
- if n is n'.+1
- then if ~~ root p x
- then sgr p.[x]
- else aux p^`() n'
- else 0
- in aux p (size p).
-
-Lemma sgp_right0 x : sgp_right 0 x = 0.
-Proof. by rewrite /sgp_right size_poly0. Qed.
-
-Lemma sgr_neighpr b p x :
- {in neighpr p x b, forall y, (sgr p.[y] = sgp_right p x)}.
-Proof.
-elim: (size p) {-2}p (leqnn (size p))=> [|n ihn] {p} p.
- rewrite leqn0 size_poly_eq0 /neighpr; move/eqP=> -> /=.
- by move=> y; rewrite next_root0 itv_xx.
-rewrite leq_eqVlt ltnS; case/orP; last exact: ihn.
-move/eqP=> sp; rewrite /sgp_right sp /=.
-case px0: root=> /=; last first.
- move=> y; rewrite /neighpr => hy /=; symmetry.
- apply: (@polyrN0_itv `[x, y]); do ?by rewrite bound_in_itv /= (itvP hy).
- move=> z; rewrite (@itv_splitU _ x true) ?bound_in_itv /= ?(itvP hy) //.
- rewrite itv_xx /=; case/predU1P=> hz; first by rewrite hz px0.
- rewrite (@next_noroot p x b) //.
- by apply: subitvPr hz=> /=; rewrite (itvP hy).
-have <-: size p^`() = n by rewrite size_deriv sp.
-rewrite -/(sgp_right p^`() x).
-move=> y; rewrite /neighpr=> hy /=.
-case: (@neighpr_wit (p * p^`()) x b)=> [||m hm].
-* case: next_rootP hy; first by rewrite itv_xx.
- by move=> ? ? ?; move/itvP->.
- by move=> c p0 -> _; case: maxrP=> _; rewrite ?itv_xx //; move/itvP->.
-* rewrite mulf_neq0 //.
- by move/eqP:sp; apply: contraTneq=> ->; rewrite size_poly0.
- (* Todo : a lemma for this *)
- move: (size_deriv p); rewrite sp /=; move/eqP; apply: contraTneq=> ->.
- rewrite size_poly0; apply: contraTneq px0=> hn; rewrite -hn in sp.
- by move/eqP: sp; case/size_poly1P=> c nc0 ->; rewrite rootC.
-* move: hm; rewrite neighpr_mul /neighpr inE /=; case/andP=> hmp hmp'.
- rewrite (polyrN0_itv _ hmp) //; last exact: next_noroot.
- rewrite (@ders0r p x m (mid x m)) ?(eqP px0) ?mid_in_itv ?bound_in_itv //;
- rewrite /= ?(itvP hmp) //; last first.
- move=> u hu /=; rewrite (@next_noroot _ x b) //.
- by apply: subitvPr hu; rewrite /= (itvP hmp').
- rewrite ihn ?size_deriv ?sp /neighpr //.
- by rewrite (subitvP _ (@mid_in_itv _ true true _ _ _)) //= ?lerr (itvP hmp').
-Qed.
-
-Lemma sgr_neighpl a p x :
- {in neighpl p a x, forall y,
- (sgr p.[y] = (-1) ^+ (odd (\mu_x p)) * sgp_right p x)
- }.
-Proof.
-elim: (size p) {-2}p (leqnn (size p))=> [|n ihn] {p} p.
- rewrite leqn0 size_poly_eq0 /neighpl; move/eqP=> -> /=.
- by move=> y; rewrite prev_root0 itv_xx.
-rewrite leq_eqVlt ltnS; case/orP; last exact: ihn.
-move/eqP=> sp; rewrite /sgp_right sp /=.
-case px0: root=> /=; last first.
- move=> y; rewrite /neighpl => hy /=; symmetry.
- move: (negbT px0); rewrite -mu_gt0; last first.
- by apply: contraFN px0; move/eqP->; rewrite rootC.
- rewrite -leqNgt leqn0; move/eqP=> -> /=; rewrite expr0 mul1r.
- symmetry; apply: (@polyrN0_itv `[y, x]);
- do ?by rewrite bound_in_itv /= (itvP hy).
- move=> z; rewrite (@itv_splitU _ x false) ?bound_in_itv /= ?(itvP hy) //.
- rewrite itv_xx /= orbC; case/predU1P=> hz; first by rewrite hz px0.
- rewrite (@prev_noroot p a x) //.
- by apply: subitvPl hz=> /=; rewrite (itvP hy).
-have <-: size p^`() = n by rewrite size_deriv sp.
-rewrite -/(sgp_right p^`() x).
-move=> y; rewrite /neighpl=> hy /=.
-case: (@neighpl_wit (p * p^`()) a x)=> [||m hm].
-* case: prev_rootP hy; first by rewrite itv_xx.
- by move=> ? ? ?; move/itvP->.
- by move=> c p0 -> _; case: minrP=> _; rewrite ?itv_xx //; move/itvP->.
-* rewrite mulf_neq0 //.
- by move/eqP:sp; apply: contraTneq=> ->; rewrite size_poly0.
- (* Todo : a lemma for this *)
- move: (size_deriv p); rewrite sp /=; move/eqP; apply: contraTneq=> ->.
- rewrite size_poly0; apply: contraTneq px0=> hn; rewrite -hn in sp.
- by move/eqP: sp; case/size_poly1P=> c nc0 ->; rewrite rootC.
-* move: hm; rewrite neighpl_mul /neighpl inE /=; case/andP=> hmp hmp'.
- rewrite (polyrN0_itv _ hmp) //; last exact: prev_noroot.
- rewrite (@ders0l p m x (mid m x)) ?(eqP px0) ?mid_in_itv ?bound_in_itv //;
- rewrite /= ?(itvP hmp) //; last first.
- move=> u hu /=; rewrite (@prev_noroot _ a x) //.
- by apply: subitvPl hu; rewrite /= (itvP hmp').
- rewrite ihn ?size_deriv ?sp /neighpl //; last first.
- by rewrite (subitvP _ (@mid_in_itv _ true true _ _ _)) //=
- ?lerr (itvP hmp').
- rewrite mu_deriv // odd_sub ?mu_gt0 //=; last by rewrite -size_poly_eq0 sp.
- by rewrite signr_addb /= mulrN1 mulNr opprK.
-Qed.
-
-Lemma sgp_right_deriv (p : {poly R}) x :
- root p x -> sgp_right p x = sgp_right (p^`()) x.
-Proof.
-elim: (size p) {-2}p (erefl (size p)) x => {p} [p|sp hp p hsp x].
- by move/eqP; rewrite size_poly_eq0; move/eqP=> -> x _; rewrite derivC.
-by rewrite /sgp_right size_deriv hsp /= => ->.
-Qed.
-
-Lemma sgp_rightNroot (p : {poly R}) x :
- ~~ root p x -> sgp_right p x = sgr p.[x].
-Proof.
-move=> nrpx; rewrite /sgp_right; case hsp: (size _)=> [|sp].
- by move/eqP:hsp; rewrite size_poly_eq0; move/eqP->; rewrite hornerC sgr0.
-by rewrite nrpx.
-Qed.
-
-Lemma sgp_right_mul p q x : sgp_right (p * q) x = sgp_right p x * sgp_right q x.
-Proof.
-case: (altP (q =P 0))=> q0; first by rewrite q0 /sgp_right !(size_poly0,mulr0).
-case: (altP (p =P 0))=> p0; first by rewrite p0 /sgp_right !(size_poly0,mul0r).
-case: (@neighpr_wit (p * q) x (1 + x))=> [||m hpq]; do ?by rewrite mulf_neq0.
- by rewrite ltr_spaddl ?ltr01.
-rewrite -(@sgr_neighpr (1 + x) _ _ m) //.
-move: hpq; rewrite neighpr_mul inE /=; case/andP=> hp hq.
-by rewrite hornerM sgrM !(@sgr_neighpr (1 + x) _ x) /neighpr.
-Qed.
-
-Lemma sgp_right_scale c p x : sgp_right (c *: p) x = sgr c * sgp_right p x.
-Proof.
-case c0: (c == 0); first by rewrite (eqP c0) scale0r sgr0 mul0r sgp_right0.
-by rewrite -mul_polyC sgp_right_mul sgp_rightNroot ?hornerC ?rootC ?c0.
-Qed.
-
-Lemma sgp_right_square p x : p != 0 -> sgp_right p x * sgp_right p x = 1.
-Proof.
-move=> np0; case: (@neighpr_wit p x (1 + x))=> [||m hpq] //.
- by rewrite ltr_spaddl ?ltr01.
-rewrite -(@sgr_neighpr (1 + x) _ _ m) //.
-by rewrite -expr2 sqr_sg (@next_noroot _ x (1 + x)).
-Qed.
-
-Lemma sgp_right_rec p x :
- sgp_right p x =
- (if p == 0 then 0 else if ~~ root p x then sgr p.[x] else sgp_right p^`() x).
-Proof.
-rewrite /sgp_right; case hs: size => [|s]; rewrite -size_poly_eq0 hs //=.
-by rewrite size_deriv hs.
-Qed.
-
-Lemma sgp_right_addp0 (p q : {poly R}) x :
- q != 0 -> (\mu_x p > \mu_x q)%N -> sgp_right (p + q) x = sgp_right q x.
-Proof.
-move=> nq0; move hm: (\mu_x q)=> m.
-elim: m p q nq0 hm => [|mq ihmq] p q nq0 hmq; case hmp: (\mu_x p)=> // [mp];
- do[
- rewrite ltnS=> hm;
- rewrite sgp_right_rec {1}addrC;
- rewrite GRing.Theory.addr_eq0]. (* Todo : fix this ! *)
- case: (altP (_ =P _))=> hqp.
- move: (nq0); rewrite {1}hqp oppr_eq0=> np0.
- rewrite sgp_right_rec (negPf nq0) -mu_gt0 // hmq /=.
- apply/eqP; rewrite eq_sym hqp hornerN sgrN.
- by rewrite oppr_eq0 sgr_eq0 -[_ == _]mu_gt0 ?hmp.
- rewrite rootE hornerD.
- have ->: p.[x] = 0.
- apply/eqP; rewrite -[_ == _]mu_gt0 ?hmp //.
- by move/eqP: hmp; apply: contraTneq=> ->; rewrite mu0.
- symmetry; rewrite sgp_right_rec (negPf nq0) add0r.
- by rewrite -/(root _ _) -mu_gt0 // hmq.
-case: (altP (_ =P _))=> hqp.
- by move: hm; rewrite -ltnS -hmq -hmp hqp mu_opp ltnn.
-have px0: p.[x] = 0.
- apply/rootP; rewrite -mu_gt0 ?hmp //.
- by move/eqP: hmp; apply: contraTneq=> ->; rewrite mu0.
-have qx0: q.[x] = 0 by apply/rootP; rewrite -mu_gt0 ?hmq //.
-rewrite rootE hornerD px0 qx0 add0r eqxx /=; symmetry.
-rewrite sgp_right_rec rootE (negPf nq0) qx0 eqxx /=.
-rewrite derivD ihmq // ?mu_deriv ?rootE ?px0 ?qx0 ?hmp ?hmq ?subn1 //.
-apply: contra nq0; rewrite -size_poly_eq0 size_deriv.
-case hsq: size=> [|sq] /=.
- by move/eqP: hsq; rewrite size_poly_eq0.
-move/eqP=> sq0; move/eqP: hsq qx0; rewrite sq0; case/size_poly1P=> c c0 ->.
-by rewrite hornerC; move/eqP; rewrite (negPf c0).
-Qed.
-
-End SignRight.
-
-(* redistribute some of what follows with in the file *)
-Section PolyRCFPdiv.
-Import Pdiv.Ring Pdiv.ComRing.
-
-Lemma sgp_rightc (x c : R) : sgp_right c%:P x = sgr c.
-Proof.
-rewrite /sgp_right size_polyC.
-case cn0: (_ == 0)=> /=; first by rewrite (eqP cn0) sgr0.
-by rewrite rootC hornerC cn0.
-Qed.
-
-Lemma sgp_right_eq0 (x : R) p : (sgp_right p x == 0) = (p == 0).
-Proof.
-case: (altP (p =P 0))=> p0; first by rewrite p0 sgp_rightc sgr0 eqxx.
-rewrite /sgp_right.
-elim: (size p) {-2}p (erefl (size p)) p0=> {p} [|sp ihsp] p esp p0.
- by move/eqP:esp; rewrite size_poly_eq0 (negPf p0).
-rewrite esp /=; case px0: root=> //=; rewrite ?sgr_cp0 ?px0//.
-have hsp: sp = size p^`() by rewrite size_deriv esp.
-rewrite hsp ihsp // -size_poly_eq0 -hsp; apply/negP; move/eqP=> sp0.
-move: px0; rewrite root_factor_theorem.
-by move=> /rdvdp_leq // /(_ p0); rewrite size_XsubC esp sp0.
-Qed.
-
-(* :TODO: backport to polydiv *)
-Lemma lc_expn_rscalp_neq0 (p q : {poly R}): lead_coef q ^+ rscalp p q != 0.
-Proof.
-case: (eqVneq q 0) => [->|nzq]; last by rewrite expf_neq0 ?lead_coef_eq0.
-by rewrite /rscalp unlock /= eqxx /= expr0 oner_neq0.
-Qed.
-Notation lcn_neq0 := lc_expn_rscalp_neq0.
-
-Lemma sgp_right_mod p q x : (\mu_x p < \mu_x q)%N ->
- sgp_right (rmodp p q) x = (sgr (lead_coef q)) ^+ (rscalp p q) * sgp_right p x.
-Proof.
-move=> mupq; case p0: (p == 0).
- by rewrite (eqP p0) rmod0p !sgp_right0 mulr0.
-have qn0 : q != 0.
- by apply/negP; move/eqP=> q0; rewrite q0 mu0 ltn0 in mupq.
-move/eqP: (rdivp_eq q p).
-rewrite eq_sym (can2_eq (addKr _ ) (addNKr _)); move/eqP->.
-case qpq0: ((rdivp p q) == 0).
- by rewrite (eqP qpq0) mul0r oppr0 add0r sgp_right_scale // sgrX.
-rewrite sgp_right_addp0 ?sgp_right_scale ?sgrX //.
- by rewrite scaler_eq0 negb_or p0 lcn_neq0.
-rewrite mu_mulC ?lcn_neq0 // mu_opp mu_mul ?mulf_neq0 ?qpq0 //.
-by rewrite ltn_addl.
-Qed.
-
-Lemma rootsC (a b c : R) : roots c%:P a b = [::].
-Proof.
-case: (altP (c =P 0))=> hc; first by rewrite hc roots0.
-by apply: no_root_roots=> x hx; rewrite rootC.
-Qed.
-
-Lemma rootsZ a b c p : c != 0 -> roots (c *: p) a b = roots p a b.
-Proof.
-have [->|p_neq0 c_neq0] := eqVneq p 0; first by rewrite scaler0.
-by apply/roots_eq => [||x axb]; rewrite ?scaler_eq0 ?(negPf c_neq0) ?rootZ.
-Qed.
-
-Lemma root_bigrgcd (x : R) (ps : seq {poly R}) :
- root (\big[(@rgcdp _)/0]_(p <- ps) p) x = all (root^~ x) ps.
-Proof.
-elim: ps; first by rewrite big_nil root0.
-move=> p ps ihp; rewrite big_cons /=.
-by rewrite (eqp_root (eqp_rgcd_gcd _ _)) root_gcd ihp.
-Qed.
-
-Definition rootsR p := roots p (- cauchy_bound p) (cauchy_bound p).
-
-Lemma roots_on_rootsR p : p != 0 -> roots_on p `]-oo, +oo[ (rootsR p).
-Proof.
-rewrite /rootsR => p_neq0 x /=; rewrite -roots_on_roots // andbC.
-by have [/(cauchy_boundP p_neq0) /=|//] := altP rootP; rewrite ltr_norml.
-Qed.
-
-Lemma rootsR0 : rootsR 0 = [::]. Proof. exact: roots0. Qed.
-
-Lemma rootsRC c : rootsR c%:P = [::]. Proof. exact: rootsC. Qed.
-
-Lemma rootsRP p a b :
- {in `]-oo, a], noroot p} -> {in `[b , +oo[, noroot p} ->
- roots p a b = rootsR p.
-Proof.
-move=> rpa rpb.
-have [->|p_neq0] := eqVneq p 0; first by rewrite rootsR0 roots0.
-apply: (eq_sorted_irr (@ltr_trans _)); rewrite ?sorted_roots // => x.
-rewrite -roots_on_rootsR -?roots_on_roots //=.
-have [rpx|] := boolP (root _ _); rewrite ?(andbT, andbF) //.
-apply: contraLR rpx; rewrite inE negb_and -!lerNgt.
-by move=> /orP[/rpa //|xb]; rewrite rpb // inE andbT.
-Qed.
-
-Lemma sgp_pinftyP x (p : {poly R}) :
- {in `[x , +oo[, noroot p} ->
- {in `[x, +oo[, forall y, sgr p.[y] = sgp_pinfty p}.
-Proof.
-rewrite /sgp_pinfty; wlog lp_gt0 : x p / lead_coef p > 0 => [hwlog|rpx y Hy].
- have [|/(hwlog x p) //|/eqP] := ltrgtP (lead_coef p) 0; last first.
- by rewrite lead_coef_eq0 => /eqP -> ? ? ?; rewrite lead_coef0 horner0.
- rewrite -[p]opprK lead_coef_opp oppr_cp0 => /(hwlog x _) Hp HNp y Hy.
- by rewrite hornerN !sgrN Hp => // z /HNp; rewrite rootN.
-have [z Hz] := poly_pinfty_gt_lc lp_gt0.
-have {Hz} Hz u : u \in `[z, +oo[ -> Num.sg p.[u] = 1.
- by rewrite inE andbT => /Hz pu_ge1; rewrite gtr0_sg // (ltr_le_trans lp_gt0).
-rewrite (@polyrN0_itv _ _ rpx (maxr y z)) ?inE ?ler_maxr ?(itvP Hy) //.
-by rewrite Hz ?gtr0_sg // inE ler_maxr lerr orbT.
-Qed.
-
-Lemma sgp_minftyP x (p : {poly R}) :
- {in `]-oo, x], noroot p} ->
- {in `]-oo, x], forall y, sgr p.[y] = sgp_minfty p}.
-Proof.
-move=> rpx y Hy; rewrite -sgp_pinfty_sym.
-have -> : p.[y] = (p \Po -'X).[-y] by rewrite horner_comp !hornerE opprK.
-apply: (@sgp_pinftyP (- x)); last by rewrite inE ler_opp2 (itvP Hy).
-by move=> z Hz /=; rewrite root_comp !hornerE rpx // inE ler_oppl (itvP Hz).
-Qed.
-
-Lemma odd_poly_root (p : {poly R}) : ~~ odd (size p) -> {x | root p x}.
-Proof.
-move=> size_p_even.
-have [->|p_neq0] := altP (p =P 0); first by exists 0; rewrite root0.
-pose b := cauchy_bound p.
-have [] := @ivt_sign p (-b) b; last by move=> x _; exists x.
- by rewrite ge0_cp // ?cauchy_bound_ge0.
-rewrite (sgp_minftyP (le_cauchy_bound p_neq0)) ?bound_in_itv //.
-rewrite (sgp_pinftyP (ge_cauchy_bound p_neq0)) ?bound_in_itv //.
-move: size_p_even; rewrite polySpred //= negbK /sgp_minfty -signr_odd => ->.
-by rewrite expr1 mulN1r sgrN mulNr -expr2 sqr_sg lead_coef_eq0 p_neq0.
-Qed.
-End PolyRCFPdiv.
-
-End PolyRCF.