#include <stddef.h>

/*@ predicate monotone_slice(int* a, size_t low, size_t up) =
  (\forall integer i,j; low <= i < j < up ==> a[i] < a[j]) ||
  (\forall integer i,j; low <= i <= j < up ==> a[i] >= a[j]);

*/

/*@
    requires TODO;
    assigns cutpoints[0 .. length];
    ensures TODO;
*/
size_t monotonic(int* a, size_t length, size_t* cutpoints) {
  cutpoints[0] = 0;
  if (length == 0) return 1;
  size_t x = 0, y = 1;
  size_t res = 1;
  /*@
    loop invariant outer_bound: 1 <= y <= length;
    loop invariant x_1: y < length ==> x == y - 1;
    loop invariant x_2: y == length ==> x == y - 1 || x == y;
    loop invariant monotone:
      \forall integer i; 0 <= i < res - 1 ==>
      monotone_slice(a,cutpoints[i],cutpoints[i+1]);
    loop invariant res_bounds: 1 <= res <= y;
    loop invariant content_bounds:
      \forall integer i; 0 <= i < res ==> 0<= cutpoints[i] <= length;
    loop invariant cuts_increase:
      \forall integer i, j; 0 <= i <= j < res ==> cutpoints[i] <= cutpoints[j];
    loop invariant last_write: cutpoints[res - 1] == x;
    loop assigns x,y,res,cutpoints[1 .. length];
    loop variant length - y;
   */
  while (y < length) {
    int increasing = a[x] < a[y];
    /*@
      loop invariant inner_bound: x + 1 <= y <= length;
      loop invariant res_low: res <= y;
      loop invariant mono_slice_1:
      increasing ==> \forall integer i, j; x <= i < j < y ==> a[i] < a[j];
      loop invariant mono_slice_2:
      (!increasing) ==> \forall integer i, j; x <= i <= j < y ==> a[i] >= a[j];
      loop assigns y;
      loop variant length - y;
    */
    while (y < length && (a[y-1] < a[y]) == increasing) y++;
    /*@ assert mono: monotone_slice(a,x,y); */
    cutpoints[res] = y;
    res++;
    /*@ assert mono_res: monotone_slice(a,cutpoints[res-2],cutpoints[res-1]);*/
    x = y;
    if (y < length) y++;
  }
  if (x < length) {
    /*@ assert last: x == length - 1; */
    /*@ assert mono_2: monotone_slice(a,x,length); */
    cutpoints[res] = length;
    res++;
    /*@ assert mono_3: monotone_slice(a,cutpoints[res - 2], cutpoints[res - 1]); */
  }
  return res;
}
