from typing import List, Tuple, TypeVar, Callable, Union
import numpy as np
import copy

T = TypeVar('T')
COLOR = TypeVar('COLOR', bound=int)
DIM = TypeVar('COLOR', bound=int)

class Pixel:
    def __init__(self, x, y, c):
        self.x = x
        self.y = y
        self.c = c

class Grid:

    def __init__(self, cells, ul_x=0, ul_y=0, prev_width=None, prev_height=None, prev_ul_x=None, prev_ul_y=None):
        self.height = len(cells)
        self.width = len(cells[0])

        if prev_height is None:
            self.orig_height = self.height
        else:
            self.orig_height = prev_height

        if prev_width is None:
            self.orig_width = self.width
        else:
            self.orig_width = prev_width

        if prev_ul_x is None:
            self.orig_ul_x = ul_x
        else:
            self.orig_ul_x = prev_ul_x

        if prev_ul_y is None:
            self.orig_ul_y = ul_y
        else:
            self.orig_ul_y = prev_ul_y

        self.ul_x = int(ul_x)
        self.ul_y = int(ul_y)

        self.pixels = self.from_grid(cells)  # pixels is a list of [(x, y, color)]

    def from_grid(self, cells):
        pixels = []
        for y, row in enumerate(cells):
            for x, color in enumerate(row):
                pixels.append((int(x), int(y), int(color)))
        return pixels

    @property
    def cells(self):
        return self.to_grid()
    
    def to_grid(self):
        if not self.pixels:
            return tuple()

        # Find the maximum x and y coordinates
        max_x = int(max(pixel[0] for pixel in self.pixels))
        max_y = int(max(pixel[1] for pixel in self.pixels))

        # Create a 2D list filled with zeros (black)
        grid = np.zeros((max_y+1, max_x+1))

        # Fill in the colors from self.pixels
        for x, y, color in self.pixels:
            grid[y, x] = color

        # Convert the 2D list to a tuple of tuples
        return tuple(tuple(row) for row in grid)

    def get_shifted_cells(self):
        # Get original cells and dimensions
        cells = self.cells
        width = self.width
        height = self.height
        
        # Get shift amounts
        x_shift = self.ul_x
        y_shift = self.ul_y
        
        # Create empty grid filled with background color (0)
        result = [[0 for _ in range(width)] for _ in range(height)]
        
        # Copy cells to shifted position
        for y in range(len(cells)):
            if y + y_shift < 0 or y + y_shift >= height:
                continue
            for x in range(len(cells[y])):
                if x + x_shift < 0 or x + x_shift >= width:
                    continue
                result[y + y_shift][x + x_shift] = cells[y][x]
                
        # Convert to tuple of tuples
        return tuple(tuple(row) for row in result)

    def __str__(self):
        """
        Returns a string representation of the Grid instance.
        """
        header = f"Upper-left coords: ({self.ul_x}, {self.ul_y})\n"
        header += f"Height: {self.height}, Width: {self.width}\n"
        header += "Cells:\n"

        # Convert cells to a formatted string, ensuring integer display
        cells_str = '\n'.join([' '.join(map(lambda x: str(int(x)), row)) for row in self.cells])
        return header + cells_str

    def __repr__(self):
        """
        Returns a string representation of the Grid instance.
        This method is used when the object is represented in the interactive shell.
        """
        return self.__str__()

def inverse_lookup(idx):
    for key, val in prim_indices.items():
        if val == idx:
            return key
        
    return None

# =================================================================== Primitives and their associated indices ===================================================================

prim_indices = {
    # Integer constants
    '0': 0,
    '1': 1,
    '2': 2,
    '3': 3,
    '4': 4,
    '5': 5,
    '6': 6,
    '7': 7,
    '8': 8,
    '9': 9,

    # Main functional primitives
    'identity': 10, 
    'get_objects1': 11,
    'colorSet': 12,
    'equal': 13,
    'not_equal': 14,
    'switch': 15,
    'index': 16,
    'add': 17,
    'sub': 18,
    'div': 19,
    'mul': 20,
    'crop': 21,
    'colorOf': 22,
    'set_pixels': 23,
    'range': 24,
    'del': 25,

    # Object attributes
    '.x': 26,        # PIXEL attribute
    '.y': 27,        # PIXEL attribute
    '.c': 28,        # PIXEL attribute
    '.max_x': 29,    # Grid attribute
    '.max_y': 30,    # Grid attribute
    '.width': 31,    # Grid attribute
    '.height': 32,    # Grid attribute
    '.ul_x': 33,     # Grid attribute
    '.ul_y': 34      # Grid attribute
}

# ======================================================================== Implementation of DSL ========================================================================

def get_width(g: Union[Grid, List[Grid]]) -> Union[DIM, List[DIM]]:
    if isinstance(g, Grid):
        return g.width
    else:
        widths = []
        for grid in g:
            widths.append(grid.width)
        return widths

def get_height(g: Union[Grid, List[Grid]]) -> Union[DIM, List[DIM]]:
    if isinstance(g, Grid):
        return g.height
    else:
        heights = []
        for grid in g:
            heights.append(grid.height)
        return heights

def get_x(p: Union[Pixel, List[Pixel], List[List[Pixel]]]) -> Union[COLOR, List[COLOR], List[List[COLOR]]]:
    if isinstance(p, List) and isinstance(p[0], List):
        x_value_lists = []
        for px_list in p:
            x_values = []
            for px in px_list:
                x_values.append(px.x)

            x_value_lists.append(x_values)
        return x_value_lists
    
    elif isinstance(p, List):
        x_values = []
        for px in p:
            x_values.append(px.x)
        return x_values
    else:
        return p.x

def get_y(p: Union[Pixel, List[Pixel], List[List[Pixel]]]) -> Union[COLOR, List[COLOR], List[List[COLOR]]]:
    if isinstance(p, List) and isinstance(p[0], List):
        y_value_lists = []
        for px_list in p:
            y_values = []
            for px in px_list:
                y_values.append(px.y)

            y_value_lists.append(y_values)
        return y_value_lists

    elif isinstance(p, List):
        y_values = []
        for px in p:
            y_values.append(px.y)
        return y_values
    else:
        return p.y

def get_color(p: Union[Pixel, List[Pixel], List[List[Pixel]]]) -> Union[COLOR, List[COLOR], List[List[COLOR]]]:
    if isinstance(p, List) and isinstance(p[0], List):
        c_value_lists = []
        for px_list in p:
            c_values = []
            for px in px_list:
                c_values.append(px.c)

            c_value_lists.append(c_values)
        return c_value_lists
    
    elif isinstance(p, List):
        c_values = []
        for px in p:
            c_values.append(px.c)
        return c_values
    else:
        return p.c

def get_index(list: List[T], i: int) -> Union[T, List[T]]:
    if isinstance(list[0], List):
        output = []
        for sublist in list:
            output.append(sublist[i])
        return output
    else:
        return list[i]

def addition(a: Union[int, List[int], List[List[int]]], 
             b: Union[int, List[int], List[List[int]]]) -> int:
    if isinstance(a, List) and isinstance(a[0], List) and isinstance(b, List) and isinstance(b[0], List):
        output_sum_lists = []
        for list_idx in range(len(a)):
            output_sums = []
            for elem_idx in range(len(a[list_idx])):
                output_sums.append(a[list_idx][elem_idx] + b[list_idx][elem_idx])
            
            output_sum_lists.append(output_sums)
        return output_sum_lists
    elif isinstance(a, List) and isinstance(a[0], List) and isinstance(b, List):
        output_sum_lists = []
        for list_idx in range(len(a)):
            output_sums = []
            for elem_idx in range(len(a[list_idx])):
                output_sums.append(a[list_idx][elem_idx] + b[list_idx])
            output_sum_lists.append(output_sums)
        return output_sum_lists
    elif isinstance(b, List) and isinstance(b[0], List) and isinstance(a, List):
        output_sum_lists = []
        for list_idx in range(len(b)):
            output_sums = []
            for elem_idx in range(len(b[list_idx])):
                output_sums.append(b[list_idx][elem_idx] + a[list_idx])
            output_sum_lists.append(output_sums)
        return output_sum_lists    
    elif isinstance(a, List) and isinstance(b, List):
        output_sums = []
        for idx in range(len(a)):
            output_sums.append(a[idx] + b[idx])
        return output_sums
    elif isinstance(a, List):
        output_sums = []
        for idx in range(len(a)):
            output_sums.append(a[idx] + b)
        return output_sums
    elif isinstance(b, List):
        output_sums = []
        for idx in range(len(b)):
            output_sums.append(a + b[idx])
        return output_sums
    else:
        return a + b

def subtraction(a: int, b: int) -> int:
    if isinstance(a, List) and isinstance(a[0], List) and isinstance(b, List) and isinstance(b[0], List):
        output_subs_lists = []
        for list_idx in range(len(a)):
            output_subs = []
            for elem_idx in range(len(a[list_idx])):
                result = a[list_idx][elem_idx] - b[list_idx][elem_idx]
                output_subs.append(max(0, result))
            
            output_subs_lists.append(output_subs)
        return output_subs_lists
    elif isinstance(a, List) and isinstance(a[0], List) and isinstance(b, List):
        output_sub_lists = []
        for list_idx in range(len(a)):
            output_subs = []
            for elem_idx in range(len(a[list_idx])):
                result = a[list_idx][elem_idx] - b[list_idx]
                output_subs.append(max(0, result))
            output_sub_lists.append(output_subs)
        return output_sub_lists
    elif isinstance(b, List) and isinstance(b[0], List) and isinstance(a, List):
        output_sub_lists = []
        for list_idx in range(len(b)):
            output_subs = []
            for elem_idx in range(len(b[list_idx])):
                result = a[list_idx] - b[list_idx][elem_idx]
                output_subs.append(max(0, result))
            output_sub_lists.append(output_subs)
        return output_sub_lists    
    elif isinstance(a, list) and isinstance(b, list):
        output_subs = []
        for idx in range(len(a)):
            result = a[idx] - b[idx]
            output_subs.append(max(0, result))
        return output_subs
    elif isinstance(a, list):
        output_subs = []
        for idx in range(len(a)):
            result = a[idx] - b
            output_subs.append(max(0, result))
        return output_subs
    elif isinstance(b, list):
        output_subs = []
        for idx in range(len(b)):
            result = a - b[idx]
            output_subs.append(max(0, result))
        return output_subs    
    else:
        result = a - b
        return max(0, result)

def division(a: int, b: int) -> int:
    if isinstance(a, list) and isinstance(b, list):
        output_quotients = []
        for idx in range(len(a)):
            output_quotients.append(a[idx] // b[idx])
        return output_quotients
    elif isinstance(a, list):
        output_quotients = []
        for idx in range(len(a)):
            output_quotients.append(a[idx] // b)
        return output_quotients
    elif isinstance(b, list):
        output_quotients = []
        for idx in range(len(b)):
            output_quotients.append(a // b[idx])
        return output_quotients
    else:
        return int(a // b)

def multiplication(a: int, b: int) -> int:
    if isinstance(a, list) and isinstance(b, list):
        output_products = []
        for idx in range(len(a)):
            output_products.append(a[idx] * b[idx])
        return output_products
    elif isinstance(a, list):
        output_products = []
        for idx in range(len(a)):
            output_products.append(a[idx] * b)
        return output_products
    elif isinstance(b, list):
        output_products = []
        for idx in range(len(b)):
            output_products.append(a * b[idx])
        return output_products
    else:
        return a * b

def modulo(a: int, b: int) -> int:
    if isinstance(a, list) and isinstance(b, list):
        output_mods = []
        for idx in range(len(a)):
            output_mods.append(a[idx] % b[idx])
        return output_mods
    elif isinstance(a, list):
        output_mods = []
        for idx in range(len(a)):
            output_mods.append(a[idx] % b)
        return output_mods
    elif isinstance(b, list):
        output_mods = []
        for idx in range(len(b)):
            output_mods.append(a % b[idx])
        return output_mods
    else:
        return a % b

def between(a: int, b: int, c: int) -> bool:
    if a < c and a >= b:
        return True
    else:
        return False

def equal(a, b):
    def equal1(a: int, b: int) -> bool:
        return a == b

    def equal2(a: List[int], b: int) -> List[bool]:
        output = []
        for tmp_a in a:
            output.append(tmp_a == b)

        return output

    if isinstance(a, List):
        if isinstance(a[0], List):
            output = []
            for idx in range(len(a)):
                output.append(equal2(a[idx], b[idx]))
            return output
        else:
            return equal2(a, b)
    else:
        return equal1(a, b)


def not_equal(a: int, b: int) -> bool:
    return a != b

def greater_than(a: int, b: int) -> bool:
    if a > b:
        return True
    else:
        return False

# TODO: type hinting
def switch(conditions, operations, otherwise):
    def switch_default(conditions, operations, otherwise):
        for idx, cond in enumerate(conditions):
            if cond:
                return operations[idx]
            
        return otherwise

    def process_list_switch(conditions, operations, otherwise):
        list_output = []
        for elem_idx in range(len(conditions[0])):
            inner_conditions = []
            for cond_idx in range(len(conditions)):
                inner_conditions.append(conditions[cond_idx][elem_idx])

            tmp_val = switch_default(inner_conditions, operations, otherwise)
            list_output.append(tmp_val)

        return list_output

    def process_obj_list_switch(conditions, operations, otherwise):
        # Here, conditions is: [
        #                          cond1: [
        #                              obj1: [bool1, bool2, bool3, ... boolN],
        #                              obj2: ...
        #                          ]
        #                          cond2: [
        #                              
        #                          ]
        #                       ]
        num_objects = len(conditions[0])
        list_output = []
        for obj_idx in range(num_objects):
            obj_output = []
            for elem_idx in range(len(conditions[0][obj_idx])):
                inner_conditions = []
                inner_operations = []
                for cond_idx in range(len(conditions)):
                    inner_conditions.append(conditions[cond_idx][obj_idx][elem_idx])
                    inner_operations.append(operations[cond_idx][obj_idx])

                tmp_val = switch_default(inner_conditions, inner_operations, otherwise)
                obj_output.append(tmp_val)
            list_output.append(obj_output)
        return list_output

    # Overloading for list processing
    if isinstance(conditions[0], List):
        # Here we have a list of objects per condition, or a list of values per condition.
        if isinstance(conditions[0][0], List):
            # We have a list of lists of lists, which suggests that we have a list of objects per condition, and each
            # object has a list of values associated with it (e.g. pixel-related values).
            return process_obj_list_switch(conditions, operations, otherwise)
        else:
            return process_list_switch(conditions, operations, otherwise)
    else:
        return switch_default(conditions, operations, otherwise)

def colorOf(g: Grid, x, y) -> COLOR:
    def single_grid_colorOf(g, x, y):   
        if isinstance(x, list) and isinstance(y, list):
            color_list = []
            for idx in range(len(x)):
                color_list.append(int(g.cells[y[idx]][x[idx]]))
            return color_list
        else:
            return int(g.cells[y][x])
        
    if isinstance(g, List):
        outputs = []
        for idx, tmp_g in enumerate(g):
            outputs.append(single_grid_colorOf(tmp_g, x[idx], y[idx]))
        return outputs
    else:
        return single_grid_colorOf(g, x, y)

def subgrid(g: Grid, x1: DIM, y1: DIM, x2: DIM, y2: DIM) -> Grid:
    return g.cells[y1:y2-1][x1:x2-1]

def colorSet(g: Union[Grid, List[Grid]]) -> Union[List[COLOR], List[List[COLOR]]]:
    if isinstance(g, Grid):
        pixels = [pixel[2] for pixel in g.pixels]
        colors = list(set(pixels))
        colors.sort()
        return colors
    else:
        all_colors = []
        for grid in g:
            pixels = [pixel[2] for pixel in grid.pixels]
            colors = list(set(pixels))
            colors.sort()
            all_colors.append(colors)
        return all_colors

def grid_all(g_list: List[Grid], func: Callable[[Pixel], bool]) -> List[Grid]:
    filtered = []
    for g in g_list:
        is_valid = True
        for pixel in g.pixels:
            if not func(pixel):
                is_valid = False
                break

        if is_valid:
            filtered.append(g)

    return filtered

def grid_any(g_list: List[Grid], func: Callable[[Pixel], bool]) -> List[Grid]:
    filtered = []
    for g in g_list:
        is_valid = False
        for pixel in g.pixels:
            if func(pixel):
                is_valid = True
                break

        if is_valid:
            filtered.append(g)

    return filtered

def set_pixels(target_grid: Union[Grid, List[Grid]], 
               set_x: Union[List[DIM], List[List[DIM]]], 
               set_y: Union[List[DIM], List[List[DIM]]],
               colors: Union[List[COLOR], List[List[COLOR]]]) -> Union[Grid, List[Grid]]:

    # if the target coord is out-of-bounds, extend the target grid as needed (this is especially useful for tiling tasks)
    def set_single_grid_pixels(target_grid: Grid, set_x: Union[DIM, List[DIM]], set_y: Union[DIM, List[DIM]], colors: Union[COLOR, List[COLOR]]) -> Grid:
        
        # At least one of set_x, set_y or colors must be a List and this indicates the number of elements to populate
        # for the integer cases.
        n = 0
        if isinstance(set_x, List):
            n = len(set_x)
        elif isinstance(set_y, List):
            n = len(set_y)
        elif isinstance(colors, List):
            n = len(colors)
        else:
            # ERROR: set_single_grid_pixels must have at least 1 list-type argument among set_x, set_y and colors.
            return None

        if isinstance(set_x, int) or isinstance(set_x, np.int64):
            set_x = np.ones(n) * set_x

        if isinstance(set_y, int) or isinstance(set_y, np.int64):
            set_y = np.ones(n) * set_y

        if isinstance(colors, int) or isinstance(colors, np.int64):
            colors = np.ones(n) * colors

        max_x = max(target_grid.width, max(set_x) + 1)
        max_y = max(target_grid.height, max(set_y) + 1)
        new_cells = np.zeros((max_y, max_x))
        for y in range(target_grid.height):
            for x in range(target_grid.width):
                new_cells[y][x] = target_grid.cells[y][x]

        for idx in range(n):
            x_coord = int(set_x[idx])
            y_coord = int(set_y[idx])
            color = colors[idx]

            new_cells[y_coord, x_coord] = color

        return Grid(new_cells)

    if isinstance(target_grid, List):
        output_grids = []
        for idx, grid in enumerate(target_grid):
            tmp_out = set_single_grid_pixels(grid, set_x[idx], set_y[idx], colors[idx])
            output_grids.append(tmp_out)

        return output_grids
    elif isinstance(set_x, List) and isinstance(set_x[0], List):
        # In this case, there is one Grid instance, but we have a list of lists to set.
        for list_idx in range(len(set_x)):
            target_grid = set_single_grid_pixels(target_grid, set_x[list_idx], set_y[list_idx], colors[list_idx])
        return target_grid
    else:
        return set_single_grid_pixels(target_grid, set_x, set_y, colors)

def crop(g: Union[Grid, List[Grid]], x1, y1, x2, y2) -> Grid:
    def crop_grid(g, x1, y1, x2, y2):
        new_pixels = []
        for pixel in g.pixels:
            if pixel[0] >= x1 and pixel[0] < x2:
                if pixel[1] >= y1 and pixel[1] < y2:
                    adjusted_pixel = (pixel[0] - x1, pixel[1] - y1, pixel[2])
                    new_pixels.append(adjusted_pixel)

        output_grid = copy.deepcopy(g)
        output_grid.ul_x = output_grid.ul_x + x1
        output_grid.ul_y = output_grid.ul_y + y1
        output_grid.pixels = new_pixels
        output_grid.width = max(pixel[0]+1 for pixel in new_pixels)
        output_grid.height = max(pixel[1]+1 for pixel in new_pixels)

        return output_grid

    if isinstance(g, Grid):
        return crop_grid(g, x1, y1, x2, y2)
    else:
        output_grids = []
        for g_idx, tmp_g in enumerate(g):
            if isinstance(x1, List):
                x1_inp = x1[g_idx]
            else:
                x1_inp = x1
                
            if isinstance(x2, List):
                x2_inp = x2[g_idx]
            else:
                x2_inp = x2

            if isinstance(y1, List):
                y1_inp = y1[g_idx]
            else:
                y1_inp = y1

            if isinstance(y2, List):
                y2_inp = y2[g_idx]
            else:
                y2_inp = y2

            tmp_out = crop_grid(tmp_g, x1_inp, y1_inp, x2_inp, y2_inp)
            output_grids.append(tmp_out)

        return output_grids

def set_ul(g: Grid, x: int, y: int) -> Grid:
    g.ul_x = x
    g.ul_y = y
    return g

def max_x(g: Union[Grid, List[Grid]]) -> Union[int, List[DIM]]:
    if isinstance(g, List):
        output = []
        for tmp_g in g:
            output.append(tmp_g.width - 1)
        return output
    else:
        return g.width - 1

def max_y(g: Union[Grid, List[Grid]]) -> Union[int, List[DIM]]:
    if isinstance(g, List):
        output = []
        for tmp_g in g:
            output.append(tmp_g.height - 1)
        return output
    else:
        return g.height - 1

def get_ul_x(g: Union[Grid, List[Grid]]) -> Union[DIM, List[DIM]]:
    if isinstance(g, Grid):
        return g.ul_x
    else:
        output = []
        for grid in g:
            output.append(grid.ul_x)
        return output

def get_ul_y(g: Union[Grid, List[Grid]]) -> Union[DIM, List[DIM]]:
    if isinstance(g, Grid):
        return g.ul_y
    else:
        output = []
        for grid in g:
            output.append(grid.ul_y)
        return output

def get_lr_x(g: Union[Grid, List[Grid]]) -> Union[DIM, List[DIM]]:
    if isinstance(g, Grid):
        return g.ul_x + g.width - 1
    else:
        output = []
        for grid in g:
            output.append(grid.ul_x + grid.width - 1)
        return output

def get_lr_y(g: Union[Grid, List[Grid]]) -> Union[DIM, List[DIM]]:
    if isinstance(g, Grid):
        return g.ul_y + g.height - 1
    else:
        output = []
        for grid in g:
            output.append(grid.ul_y + grid.height - 1)
        return output
    
def get_pixels(g: Union[Grid, List[Grid]]) -> Union[List[Pixel], List[List[Pixel]]]:
    if isinstance(g, Grid):
        return [Pixel(x, y, c) for x, y, c in g.pixels]
    else:
        output = []
        for grid in g:
            output.append([Pixel(x, y, c) for x, y, c in grid.pixels])
        return output

def get_range(a: int, b: int) -> List[int]:
    return np.arange(a, b+1)


# =================================================================== Number of arguments for each primitive =================================================================

arg_counts = [
    0,
    0,
    0,
    0,
    0,
    0,
    0,
    0,
    0,
    0,
    1,
    1,
    1,
    2,
    2,
    3,
    2,
    2,
    2,
    2,
    2,
    5,
    3,
    4,
    2,
    1,
    1,
    1,
    1,
    1,
    1,
    1,
    1,
    1
]

# ======================================================= Mappings from primitive name to actual function ======================================================================

semantics = {
    # Integer constants
    '0': 0,
    '1': 1,
    '2': 2,
    '3': 3,
    '4': 4,
    '5': 5,
    '6': 6,
    '7': 7,
    '8': 8,
    '9': 9,

    # Main functional primitives
    'identity': lambda x: x,
    'get_objects1': lambda x: x,        # TODO: to be implemented
    'colorSet': colorSet,
    'equal': equal,
    'not_equal': not_equal,
    'switch': switch,
    'index': get_index,
    'add': addition,
    'sub': subtraction,
    'div': division,
    'mul': multiplication,
    'crop': crop,
    'colorOf': colorOf,
    
    # Given a list of x coordinates and y coordinates for the pixels to modify in the target grid,
    # is sets those pixels' colors to the values passed as fourth argument.
    'set_pixels': set_pixels,
    'range': get_range,

    'del': lambda x: x,       # This is actually a special primitive that is implemented at the program execution level
                              # where state memroy management is possible.

    # Object attributes
    '.x': lambda pixel: get_x(pixel),
    '.y': lambda pixel: get_y(pixel),
    '.c': lambda pixel: get_color(pixel),
    '.max_x': max_x,
    '.max_y': max_y,
    '.width': get_width,
    '.height': get_height,
    '.ul_x': get_ul_x,
    '.ul_y': get_ul_y
}

pixel_attributes = ['.x', '.y', '.c']

# ============================================================================== Useful sub-routines =========================================================================

def get_subroutine_rot90(ctx_ref, start_ref):
    program = []

    program.append(('sub', [(ctx_ref+0, '.max_y'), (ctx_ref+0, '.y')]))
    program.append(('colorOf', [ctx_ref+0, (ctx_ref+0, '.x'), start_ref+1]))
    program.append(('del', [start_ref+1]))

    return program, start_ref+1

def get_subroutine_rot180(ctx_ref, start_ref):
    program = []

    program.append(('sub', [(ctx_ref+0, '.max_x'), (ctx_ref+0, '.x')]))
    program.append(('sub', [(ctx_ref+0, '.max_y'), (ctx_ref+0, '.y')]))
    program.append(('colorOf', [ctx_ref+0, start_ref+1, start_ref+2]))
    program.append(('del', [start_ref+1]))
    program.append(('del', [start_ref+1]))

    return program, start_ref+1

def get_subroutine_rot270(ctx_ref, start_ref):
    program = []

    program.append(('sub', [(ctx_ref+0, '.max_x'), (ctx_ref+0, '.x')])),
    program.append(('colorOf', [ctx_ref+0, start_ref+1, (ctx_ref+0, '.y')]))
    program.append(('del', [start_ref+1]))

    return program, start_ref+1

def get_subroutine_hmirror(ctx_ref, start_ref):
    program = []

    program.append(('sub', [(ctx_ref+0, '.max_x'), (ctx_ref+0, '.x')])),
    program.append(('colorOf', [ctx_ref+0, start_ref+1, (ctx_ref+0, '.y')]))
    program.append(('del', [start_ref+1]))

    return program, start_ref+1

def get_subroutine_vmirror(ctx_ref, start_ref):
    program = []

    program.append(('sub', [(ctx_ref+0, '.max_y'), (ctx_ref+0, '.y')])),
    program.append(('colorOf', [ctx_ref+0, (ctx_ref+0, '.x'), start_ref+1]))
    program.append(('del', [start_ref+1]))

    return program, start_ref+1
