diff options
| author | Christian C <cc@localhost> | 2025-03-05 16:14:18 -0800 | 
|---|---|---|
| committer | Christian C <cc@localhost> | 2025-03-05 16:14:18 -0800 | 
| commit | 74e46ee7709785f91e04ac42619e4a7e9f90bc18 (patch) | |
| tree | 82f5140c42cff58d9a980e2737eea808e1820560 | |
| parent | a3684826417bf9cea4c24e8a82e9574feb64576d (diff) | |
Modularize
| -rw-r--r-- | include/lib/algo/avl_tree.h | 40 | ||||
| -rw-r--r-- | include/lib/monad.h | 9 | ||||
| -rw-r--r-- | include/lib/seg/mask_data.h | 46 | ||||
| -rw-r--r-- | src/lib/algo/avl_tree.c | 90 | ||||
| -rw-r--r-- | src/lib/seg/mask_data.c | 175 | ||||
| -rw-r--r-- | src/main.c | 13 | 
6 files changed, 373 insertions, 0 deletions
| diff --git a/include/lib/algo/avl_tree.h b/include/lib/algo/avl_tree.h new file mode 100644 index 0000000..80f0c44 --- /dev/null +++ b/include/lib/algo/avl_tree.h @@ -0,0 +1,40 @@ +#ifndef INC_LIB_ALGO_AVL_TREE_H +#define INC_LIB_ALGO_AVL_TREE_H + +#include <lib/bool.h> +#include <stdint.h> +#include <stdlib.h> + +struct AVLNode { +  void* data; +  bool_t (*compare)(void*, void*); +  struct AVLNode* left; +  struct AVLNode* right; +  uint8_t height; +}; + +// Get the height of an AVL node +uint8_t get_height(struct AVLNode* node); + +// Get the Maximum Height between two +uint8_t max_height(uint8_t a, uint8_t b); + +// Get the balance factor of a node +ssize_t get_balance_factor(struct AVLNode* node); + +// Rotate an AVL node right +struct AVLNode* right_rotate(struct AVLNode* parent); + +// Rotate an AVL node left +struct AVLNode* left_rotate(struct AVLNode* parent); + +// In-order traversal print pointer +void print_in_order(struct AVLNode* root); + +// Free avl tree nodes starting at root +void free_avl_tree(struct AVLNode* root); + +// Free avl tree and their data starting at root +void free_avl_tree_nodes(struct AVLNode* root); + +#endif diff --git a/include/lib/monad.h b/include/lib/monad.h new file mode 100644 index 0000000..4a68e0a --- /dev/null +++ b/include/lib/monad.h @@ -0,0 +1,9 @@ +#ifndef INC_LIB_MONAD_H +#define INC_LIB_MONAD_H + +struct Result { +  void* data; +  bool_t success; +}; + +#endif diff --git a/include/lib/seg/mask_data.h b/include/lib/seg/mask_data.h new file mode 100644 index 0000000..595ffd6 --- /dev/null +++ b/include/lib/seg/mask_data.h @@ -0,0 +1,46 @@ +#ifndef INC_LIB_SEG_MASK_DATA_H +#define INC_LIB_SEG_MASK_DATA_H + +#include <lib/algo/avl_tree.h> +#include <lib/monad.h> + +struct MaskData { +  uint16_t label; +  size_t area; +  size_t perimeter; +}; + +// Allocate Mask Data for Label +struct MaskData* create_mask_data(uint16_t label); + +// Compare mask data labels +bool_t compare_labels(struct MaskData* left, struct MaskData* right); + +// Create AVL Mask node +struct AVLNode* create_avl_mask_node(struct MaskData* data); + +// Insert MaskData into the AVL Tree +struct Result insert_mask(struct AVLNode* node, struct MaskData* data); + +// Allocate a label's Mask data in a tree +//  If it already exists, skip the allocation +struct AVLNode* insert_mask_alloc(struct AVLNode* node, uint16_t label); + +// Print AVL Node Mask Data Label +void print_label(struct AVLNode* root); + +// Increase the label's area +bool_t increase_label_area(struct AVLNode* root, uint16_t label); + +// Increase the label's perimeter +bool_t increase_label_perimeter(struct AVLNode* root, uint16_t label); + +// Increase the label's area +//  Create an AVL node if it doesn't exist +struct AVLNode* increase_label_area_alloc(struct AVLNode* root, uint16_t label); + +// Increase the label's perimeter +//  Create an AVL node if it doesn't exist +struct AVLNode* increase_label_perimeter_alloc(struct AVLNode* root, uint16_t label); + +#endif diff --git a/src/lib/algo/avl_tree.c b/src/lib/algo/avl_tree.c new file mode 100644 index 0000000..98cd18b --- /dev/null +++ b/src/lib/algo/avl_tree.c @@ -0,0 +1,90 @@ +#include <lib/algo/avl_tree.h> +#include <lib/monad.h> + +#include <stddef.h> +#include <stdio.h> + +// Get the height of an AVL node +uint8_t get_height(struct AVLNode* node) +{ +  if (node == NULL) { +    return 0; +  } +  return node->height; +} + +// Get the Maximum Height between two +uint8_t max_height(uint8_t a, uint8_t b) +{ +  return (a > b) ? a : b; +} + +// Get the balance factor of a node +ssize_t get_balance_factor(struct AVLNode* node) +{ +  if (node == NULL) { +    return 0; +  } +  return get_height(node->left) - get_height(node->right); +} + +// Rotate an AVL node right +struct AVLNode* right_rotate(struct AVLNode* parent) +{ +  struct AVLNode* child1 = parent->left; +  struct AVLNode* child2 = child1->right; + +  child1->right = parent; +  parent->left = child2; + +  parent->height = max_height(get_height(parent->left), get_height(parent->right)) + 1; +  child1->height = max_height(get_height(child1->left), get_height(child1->right)) + 1; +  return child1; +} + +// Rotate an AVL node left +struct AVLNode* left_rotate(struct AVLNode* parent) +{ +  struct AVLNode* child1 = parent->right; +  struct AVLNode* child2 = child1->left; + +  child1->left = parent; +  parent->right = child2; + +  parent->height = max_height(get_height(parent->left), get_height(parent->right)) + 1; +  child1->height = max_height(get_height(child1->left), get_height(child1->right)) + 1; +  return child1; +} + +// In-order traversal print pointer +void print_in_order(struct AVLNode* root) +{ +  if (root != NULL) { +    print_in_order(root->left); +    printf("%p ", root->data); +    print_in_order(root->right); +  } +} + +// Free avl tree nodes starting at root +void free_avl_tree(struct AVLNode* root) +{ +  if (root != NULL) { +    free_avl_tree(root->left); +    free_avl_tree(root->right); +    free(root); +  } +} + +// Free avl tree and their data starting at root +void free_avl_tree_nodes(struct AVLNode* root) +{ +  if (root != NULL) { +    free_avl_tree_nodes(root->left); +    free_avl_tree_nodes(root->right); +    if (root->data != NULL) { +      free(root->data); +    } +    free(root); +  } +} diff --git a/src/lib/seg/mask_data.c b/src/lib/seg/mask_data.c new file mode 100644 index 0000000..00e4807 --- /dev/null +++ b/src/lib/seg/mask_data.c @@ -0,0 +1,175 @@ +#include <lib/seg/mask_data.h> + +#include <stdio.h> + +// Allocate Mask Data for Label +struct MaskData* create_mask_data(uint16_t label) +{ +  struct MaskData *data = (struct MaskData*)malloc(sizeof(struct MaskData)); +  data->label = label; +  data->area = 0; +  data->perimeter = 0; +  return data; +} + +// Compare mask data labels +bool_t compare_labels(struct MaskData* left, struct MaskData* right) +{ +  return left->label < right->label; +} + +// Create AVL Mask node +struct AVLNode* create_avl_mask_node(struct MaskData* data) +{ +  struct AVLNode* node = (struct AVLNode*)malloc(sizeof(struct AVLNode)); +  if (node == NULL) { +    return NULL; +  } +  node->data = data; +  node->compare = (bool_t (*)(void*,void*))&compare_labels; +  node->left = NULL; +  node->right = NULL; +  node->height = 1; // Leaf initially +  return node; +} + +// Insert MaskData into the AVL Tree +struct Result insert_mask(struct AVLNode* node, struct MaskData* data) +{ +  struct Result result; +  // 1. Standard BST insertion +  if (node == NULL) { +    return (struct Result) {create_avl_mask_node(data), TRUE}; +  } + +  struct MaskData *node_data = (struct MaskData*)node->data; +  if (node->compare(data, node_data)) { +    result = insert_mask(node->left, data); +    if (!result.success) { +      fprintf(stderr, "Failed to insert!"); +      return result; +    } +    node->left = (struct AVLNode*)result.data; +  } else if (node->compare(node->data, data)) { +    result = insert_mask(node->right, data); +    if (!result.success) { +      fprintf(stderr, "Failed to insert!"); +      return result; +    } +    node->right = (struct AVLNode*)result.data; +  } else { +    return (struct Result) {node, FALSE}; +  } + +  // 2. Update height of the ancestor node +  node->height = 1 + max_height(get_height(node->left), get_height(node->right)); + +  ssize_t balance = get_balance_factor(node); + +  // 4. If the node becomes unbalanced + +  // LeftLeft +  if ((balance > 1) && node->compare(data, node->left->data)) { +    return (struct Result) {right_rotate(node), TRUE}; +  } +  // RightRight +  if ((balance < -1) && node->compare(node->right->data, data)) { +    return (struct Result) {left_rotate(node), TRUE}; +  } +  // LeftRight +  if ((balance > 1) && node->compare(node->left->data, data)) { +    return (struct Result) {right_rotate(node), TRUE}; +  } +  // RightLeft +  if ((balance < -1) && node->compare(data,node->right->data)) { +    return (struct Result) {left_rotate(node), TRUE}; +  } +  return (struct Result) {node, TRUE}; +} + +// Allocate a label's Mask data in a tree +//  If it already exists, skip the allocation +struct AVLNode* insert_mask_alloc(struct AVLNode* node, uint16_t label) +{ +  struct MaskData* data = create_mask_data(label); +  struct Result result = insert_mask(node, data); +  if (!result.success) { +    free(data); +  } +  return (struct AVLNode*)result.data; +} + +// Print AVL Node Mask Data Label +void print_label(struct AVLNode* root) +{ +  if (root != NULL) { +    print_label(root->left); +    struct MaskData* data = root->data; +    printf("%d: (%zu, %zu) ", data->label, data->area, data->perimeter); +    print_label(root->right); +  } +} + +// Increase the label's area +bool_t increase_label_area(struct AVLNode* root, uint16_t label) +{ +  if (root == NULL) { +    return FALSE; +  } +  struct MaskData* data = (struct MaskData*)root->data; +  if (data->label == label) { +    data->area++; +  } +  else if (data->label > label) { +    return increase_label_area(root->left, label); +  } +  else if (data->label < label) { +    return increase_label_area(root->right, label); +  } +  return TRUE; +} + +// Increase the label's perimeter +bool_t increase_label_perimeter(struct AVLNode* root, uint16_t label) +{ +  if (root == NULL) { +    return FALSE; +  } +  struct MaskData* data = (struct MaskData*)root->data; +  if (data->label == label) { +    data->perimeter++; +  } +  else if (data->label > label) { +    return increase_label_perimeter(root->left, label); +  } +  else if (data->label < label) { +    return increase_label_perimeter(root->right, label); +  } +  return TRUE; +} + +// Increase the label's area +//  Create an AVL node if it doesn't exist +struct AVLNode* increase_label_area_alloc(struct AVLNode* root, uint16_t label) +{ +  struct AVLNode* new_root = root; +  bool_t success = increase_label_area(new_root, label); +  if (success == FALSE) { +    new_root = insert_mask_alloc(new_root, label); +    increase_label_area(new_root, label); +  } +  return new_root; +} + +// Increase the label's perimeter +//  Create an AVL node if it doesn't exist +struct AVLNode* increase_label_perimeter_alloc(struct AVLNode* root, uint16_t label) +{ +  struct AVLNode* new_root = root; +  bool_t success = increase_label_perimeter(new_root, label); +  if (success == FALSE) { +    new_root = insert_mask_alloc(new_root, label); +    increase_label_perimeter(new_root, label); +  } +  return new_root; +} @@ -31,6 +31,7 @@ int main(int argc, char** argv)    print_label(root);    printf("\n");    free_avl_tree_nodes(root); +  root = NULL;    //-----------------------------------------------    //-LIST-FILES-IN-DIRECTORY-----------------------    //----------------------------------------------- @@ -169,6 +170,18 @@ int main(int argc, char** argv)    CloseWindow();  #else    if (masks != NULL) { +    for (size_t y = 0; y < height; y++) { +      for (size_t x = 0; x < width; x++) { +	if (masks[x + y*width] != 0) { +	  root = increase_label_area_alloc(root, masks[x + y*width]); +	} +      } +    } +    printf("Inorder traversal of AVL tree: "); +    print_label(root); +    printf("\n"); +    free_avl_tree_nodes(root); +    root = NULL;      write_array("../out.bin", masks, width*height*sizeof(uint16_t));      free(masks);    } | 
