#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <math.h>
void print_pairs_u16(unsigned short *a, size_t size) {
size_t i;
for (i = 0; i < size; i += 2) {
printf("[%u, %u]", a
[i
], a
[i
+ 1]); }
}
void print_pairs_u32(unsigned int *a, size_t size) {
size_t i;
for (i = 0; i < size; i += 2) {
printf("[%u, %u]", a
[i
], a
[i
+ 1]); }
}
static inline size_t find_index_u16(unsigned short *a, size_t size, unsigned int value) {
size_t i;
for (i = 0; i < size; i++) if (a[i] == value) return i;
return size;
}
static inline size_t find_index_u32(unsigned int *a, size_t size, unsigned int value) {
size_t i;
for (i = 0; i < size; i++) if (a[i] == value) return i;
return size;
}
#define print_pairs(a, n) _Generic((a), unsigned short * : print_pairs_u16, unsigned int * : print_pairs_u32, default : print_pairs_u32)((a), (n))
#define find_index(a, n, v) _Generic((a), unsigned short * : find_index_u16, unsigned int * : find_index_u32, default : find_index_u32)((a), (n), (v))
#define array_cap(a) (sizeof (a) / sizeof *(a))
#define array_print_as_pairs(a) print_pairs((a), array_cap(a))
#define array_find_index(a, v) find_index(a, array_cap(a), v)
typedef unsigned long long type_key;
typedef unsigned short type_value_elem;
typedef struct value {
type_value_elem values[4];
} type_value;
typedef struct entry {
type_key key;
type_value value;
} type_entry;
typedef unsigned int type_index;
typedef struct table {
type_index bin_power, entry_power, n_entries, *bins;
type_entry *entries;
} type_table;
static inline type_index bins_cap(type_table *table) {
return (type_index)1 << table->bin_power;
}
static inline type_index entries_cap(type_table *table) {
return (type_index)1 << table->entry_power;
}
static inline size_t bins_n_bytes(type_table *table) {
return sizeof *table->bins * bins_cap(table);
}
static inline size_t entries_n_bytes(type_table *table) {
return sizeof *table->entries * entries_cap(table);
}
void table_inspect_bins(type_table *table) {
for (size_t i = 0; i < bins_cap(table); i++)
printf("bins[%zu]: %u\n", i
, table
->bins
[i
]); }
void table_inspect(type_table *table) {
printf("bin_power: %u\n", table
->bin_power
); printf("entry_power: %u\n", table
->entry_power
); printf("n_entries: %u\n", table
->n_entries
); printf("bins: %p\n", table
->bins
); printf("entries: %p\n", table
->entries
); printf("bins_cap: %x %d\n", bins_cap
(table
), bins_cap
(table
)); printf("entries_cap: %x %d\n", entries_cap
(table
), entries_cap
(table
)); }
type_table *table_new(type_index bin_power, type_index entry_power) {
type_table
*table
= malloc(sizeof (type_table
)); table->bin_power = bin_power;
table->entry_power = entry_power;
table->n_entries = 0;
table
->bins
= malloc(bins_n_bytes
(table
)); table
->entries
= malloc(entries_n_bytes
(table
)); memset(table
->bins
, 0, bins_n_bytes
(table
)); if (!table || !table->bins || !table->entries)
fprintf(stderr
, "table_new: out of memory\n"), exit(EXIT_FAILURE
); return table;
}
void table_free(type_table *table) {
}
static inline type_index calc_bin_index(type_table *table, type_key key) {
return (bins_cap(table) - 1) & key;
}
static inline type_index find_bin_index(type_table *table, type_key key) {
type_index found = calc_bin_index(table, key);
type_index collision = 0;
type_index borderline = bins_cap(table) - 1;
type_index *bins = table->bins;
type_entry *entries = table->entries;
while (bins[found] && entries[bins[found]].key != key) {
collision++;
if (borderline < collision)
fprintf(stderr
, "borderline: %u, collision: %u\n", borderline
, collision
), exit(EXIT_FAILURE
); else
found = calc_bin_index(table, found + 1);
}
return found;
}
static inline void table_put_at(type_table *table, type_key key, type_value value, type_index bin_index) {
type_index borderline = entries_cap(table) - 1;
type_index *bins = table->bins;
type_entry *entries = table->entries;
if (!bins[bin_index]) {
// bins[x] == 0はビンが未使用という表現。そのため entries[0] は欠番。最初のデータは entries[1] に入る。
if (borderline < table->n_entries + 1)
fprintf(stderr
, "borderline: %u, (table->n_entries + 1): %u\n", borderline
, table
->n_entries
+ 1), exit(EXIT_FAILURE
); table->n_entries++;
bins[bin_index] = table->n_entries;
entries[bins[bin_index]] = (type_entry){key, value};
} else {
// 既存のエントリに上書き。
entries[bins[bin_index]] = (type_entry){key, value};
}
}
void table_put(type_table *table, type_key key, type_value value) {
return table_put_at(table, key, value, find_bin_index(table, key));
}
static inline type_value *table_get_at(type_table *table, type_index bin_index) {
type_index *bins = table->bins;
type_entry *entries = table->entries;
return !bins[bin_index] ? 0 : &entries[bins[bin_index]].value;
}
type_value *table_get(type_table *table, type_key key) {
return table_get_at(table, find_bin_index(table, key));
}
int table_contains_key(type_table *table, type_key key) {
return !!table_get(table, key);
}
type_index table_size(type_table *table) {
return table->n_entries;
}
static inline type_value *value_append(type_value *value, type_value_elem e) {
type_index cap = array_cap(value->values), i = array_find_index(value->values, 0);
if (i < cap)
value->values[i] = e;
else
fprintf(stderr
, "value->values cap: %u, required: %u\n", cap
, cap
+ 1), exit(EXIT_FAILURE
); return value;
}
struct answer {
type_key c;
type_value_elem pairs[10];
};
struct answer answer_from(type_value *value, type_value_elem last_b, type_key c) {
struct answer ans = {c};
size_t i, cap = array_cap(value->values);
type_value_elem a, b;
for (i = 0; i <= cap; i++) {
b = i < cap ? value->values[i] : last_b;
a = round(cbrt(c + (type_key)b * b * b));
ans.pairs[i * 2 + 0] = a;
ans.pairs[i * 2 + 1] = b;
}
return ans;
}
struct answer f(unsigned m) {
type_table *table = table_new(27, 24);
type_value *value = 0;
type_index index;
type_value_elem a, b;
type_key c;
struct answer ans;
for (a = 1; ; a++) {
for (b = 1; b < a; b++) {
c = (type_key)a * a * a - (type_key)b * b * b;
index = find_bin_index(table, c);
value = table_get_at(table, index);
if (!value)
table_put_at(table, c, (type_value){b}, index);
else if (array_find_index(value->values, 0) + 1 == m)
goto finally;
else
value_append(value, b);
}
}
finally:
ans = answer_from(value, b, c);
table_free(table);
return ans;
}
int main() {
struct answer ans = f(5);
array_print_as_pairs(ans.pairs);
return 0;
}
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <math.h>
void print_pairs_u16(unsigned short *a, size_t size) {
  size_t i;
  putchar('[');
  for (i = 0; i < size; i += 2) {
    if (0 < i) printf(", ");
    printf("[%u, %u]", a[i], a[i + 1]);
  }
  putchar(']');
}
void print_pairs_u32(unsigned int *a, size_t size) {
  size_t i;
  putchar('[');
  for (i = 0; i < size; i += 2) {
    if (0 < i) printf(", ");
    printf("[%u, %u]", a[i], a[i + 1]);
  }
  putchar(']');
}
static inline size_t find_index_u16(unsigned short *a, size_t size, unsigned int value) {
  size_t i;
  for (i = 0; i < size; i++) if (a[i] == value) return i;
  return size;
}
static inline size_t find_index_u32(unsigned int *a, size_t size, unsigned int value) {
  size_t i;
  for (i = 0; i < size; i++) if (a[i] == value) return i;
  return size;
}
#define print_pairs(a, n) _Generic((a), unsigned short * : print_pairs_u16, unsigned int * : print_pairs_u32, default : print_pairs_u32)((a), (n))
#define find_index(a, n, v) _Generic((a), unsigned short * : find_index_u16, unsigned int * : find_index_u32, default : find_index_u32)((a), (n), (v))
#define array_cap(a) (sizeof (a) / sizeof *(a))
#define array_print_as_pairs(a) print_pairs((a), array_cap(a))
#define array_find_index(a, v) find_index(a, array_cap(a), v)

typedef unsigned long long type_key;
typedef unsigned short type_value_elem;
typedef struct value {
  type_value_elem values[4];
} type_value;
typedef struct entry {
  type_key key;
  type_value value;
} type_entry;
typedef unsigned int type_index;
typedef struct table {
  type_index bin_power, entry_power, n_entries, *bins;
  type_entry *entries;
} type_table;
static inline type_index bins_cap(type_table *table) {
  return (type_index)1 << table->bin_power;
}
static inline type_index entries_cap(type_table *table) {
  return (type_index)1 << table->entry_power;
}
static inline size_t bins_n_bytes(type_table *table) {
  return sizeof *table->bins * bins_cap(table);
}
static inline size_t entries_n_bytes(type_table *table) {
  return sizeof *table->entries * entries_cap(table);
}
void table_inspect_bins(type_table *table) {
  for (size_t i = 0; i < bins_cap(table); i++)
    printf("bins[%zu]: %u\n", i, table->bins[i]);
}
void table_inspect(type_table *table) {
  printf("table: %p\n", table);
  printf("bin_power: %u\n", table->bin_power);
  printf("entry_power: %u\n", table->entry_power);
  printf("n_entries: %u\n", table->n_entries);
  printf("bins: %p\n", table->bins);
  printf("entries: %p\n", table->entries);
  printf("bins_cap: %x %d\n", bins_cap(table), bins_cap(table));
  printf("entries_cap: %x %d\n", entries_cap(table), entries_cap(table));
}
type_table *table_new(type_index bin_power, type_index entry_power) {
  type_table *table = malloc(sizeof (type_table));
  table->bin_power = bin_power;
  table->entry_power = entry_power;
  table->n_entries = 0;
  table->bins = malloc(bins_n_bytes(table));
  table->entries = malloc(entries_n_bytes(table));
  memset(table->bins, 0, bins_n_bytes(table));
  if (!table || !table->bins || !table->entries)
    fprintf(stderr, "table_new: out of memory\n"), exit(EXIT_FAILURE);
  return table;
}
void table_free(type_table *table) {
  free(table->bins), free(table->entries), free(table);
}
static inline type_index calc_bin_index(type_table *table, type_key key) {
  return (bins_cap(table) - 1) & key;
}
static inline type_index find_bin_index(type_table *table, type_key key) {
  type_index found = calc_bin_index(table, key);
  type_index collision = 0;
  type_index borderline = bins_cap(table) - 1;
  type_index *bins = table->bins;
  type_entry *entries = table->entries;
  while (bins[found] && entries[bins[found]].key != key) {
    collision++;
    if (borderline < collision)
      fprintf(stderr, "borderline: %u, collision: %u\n", borderline, collision), exit(EXIT_FAILURE);
    else
      found = calc_bin_index(table, found + 1);
  }
  return found;
}
static inline void table_put_at(type_table *table, type_key key, type_value value, type_index bin_index) {
  type_index borderline = entries_cap(table) - 1;
  type_index *bins = table->bins;
  type_entry *entries = table->entries;
  if (!bins[bin_index]) {
    // bins[x] == 0はビンが未使用という表現。そのため entries[0] は欠番。最初のデータは entries[1] に入る。
    if (borderline < table->n_entries + 1)
      fprintf(stderr, "borderline: %u, (table->n_entries + 1): %u\n", borderline, table->n_entries + 1), exit(EXIT_FAILURE);
    table->n_entries++;
    bins[bin_index] = table->n_entries;
    entries[bins[bin_index]] = (type_entry){key, value};
  } else {
    // 既存のエントリに上書き。
    entries[bins[bin_index]] = (type_entry){key, value};
  }
}
void table_put(type_table *table, type_key key, type_value value) {
  return table_put_at(table, key, value, find_bin_index(table, key));
}
static inline type_value *table_get_at(type_table *table, type_index bin_index) {
  type_index *bins = table->bins;
  type_entry *entries = table->entries;
  return !bins[bin_index] ? 0 : &entries[bins[bin_index]].value;
}
type_value *table_get(type_table *table, type_key key) {
  return table_get_at(table, find_bin_index(table, key));
}
int table_contains_key(type_table *table, type_key key) {
  return !!table_get(table, key);
}
type_index table_size(type_table *table) {
  return table->n_entries;
}

static inline type_value *value_append(type_value *value, type_value_elem e) {
  type_index cap = array_cap(value->values), i = array_find_index(value->values, 0);
  if (i < cap)
    value->values[i] = e;
  else
    fprintf(stderr, "value->values cap: %u, required: %u\n", cap, cap + 1), exit(EXIT_FAILURE);
  return value;
}
struct answer {
  type_key c;
  type_value_elem pairs[10];
};
struct answer answer_from(type_value *value, type_value_elem last_b, type_key c) {
  struct answer ans = {c};
  size_t i, cap = array_cap(value->values);
  type_value_elem a, b;
  for (i = 0; i <= cap; i++) {
    b = i < cap ? value->values[i] : last_b;
    a = round(cbrt(c + (type_key)b * b * b));
    ans.pairs[i * 2 + 0] = a;
    ans.pairs[i * 2 + 1] = b;
  }
  return ans;
}
struct answer f(unsigned m) {
  type_table *table = table_new(27, 24);
  type_value *value = 0;
  type_index index;
  type_value_elem a, b;
  type_key c;
  struct answer ans;
  for (a = 1; ; a++) {
    for (b = 1; b < a; b++) {
      c = (type_key)a * a * a - (type_key)b * b * b;
      index = find_bin_index(table, c);
      value = table_get_at(table, index);
      if (!value) 
        table_put_at(table, c, (type_value){b}, index);
      else if (array_find_index(value->values, 0) + 1 == m) 
        goto finally;
      else 
        value_append(value, b);
    }
  }
 finally:
  ans = answer_from(value, b, c);
  table_free(table);
  return ans;
}
int main() {
  struct answer ans = f(5);
  array_print_as_pairs(ans.pairs);
  return 0;
}
