Skip to content

Header-only, type-safe, generic data structures and algorithms for C.

License

Notifications You must be signed in to change notification settings

prathameshd30/prime-headers

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

2 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Prime Headers

Header-only, type-safe, generic data structures and algorithms for C.

Multiplatform CMake Build, CTest, Valgrind Profiling

Prime Headers is (aiming to be) a headers-only, easy to use, type-safe way to use generic data structures and algorithms in C Projects.

It aims to have each data structure and algorithm thoroughly tested with primitive and pointer-to-structs examples.

It works primarily by abusing the C preprocessor to a scale hitherto undreamt of (I have later realised that this scale was indeed dreamt to).

It uses macros with arguments to generate the appropriate structs and functions at the preprocessing stage, leading to no runtime overhead. However, this approach has definite drawbacks, which are clarified further.

It uses CMake as the build system, Valgrind for checking memory safety (which needs to be migrated to AddressSanitizer as Valgrind is too slow) and GitHub Actions for checking cross-platform compatibility.

Features

  • Header-only, generic, type-safe data structures and algorithms.
  • Build tested on multiple platforms and compilers using CMake and GitHub actions.
  • Unit tested using CTest.
  • Memory profiled using Valgrind.
  • Supports primitive data structures as well as pointer-to-structs.

Currently Tested Platforms

  • Ubuntu
    • gcc
    • clang
  • Windows
    • cl (MSVC)

To Do

  • Use GitHub Actions for testing
  • Use Valgrind for testing
  • Setup CMake, CTest and GitHub actions
  • Migrate memory profiler to AddressSanitizer from Valgrind
  • Write primitive as well as pointer-to-structs examples for each data structure and algorithm

Checklist

  • Each function, with pointer parameters, should check for NULL for each pointer parameter.
  • Have a thorough deletion mechanism for each data structure.
  • Library functions in snake_case_##DATA_STRUCTURE()..
  • All reallocs need to be first assigned to temp variables, so as to not lose access to data.

To build and test locally

After ensuring you have the prerequisite dependencies (CMake and Valgrind), run these commands in the root directory of the repository.

$ pwd # Make sure you are in the root directory of the repository
$ mkdir build
$ cd build
$ cmake ..
$ cmake --build .
$ ctest # Just run the tests without memory profiling
$ ctest -T memcheck -C Debug --test-dir ../build # Valgrind memcheck and test

Data structures and algorithms implemented

The project is a work in progress, with an emphasis on getting things working first. Apologies for the ugly/(currently wrong) code.

  • Vector
  • Stack using Vector
  • Stack using Linked List
  • Queue using Vector
  • Queue using Linked List
  • Deque (Double Ended Queue) using Vector
  • Deque (Double Ended Queue) using Doubly Linked List
  • Singly Linked List
  • Set
  • Unordered Set
  • Map
  • Unordered Map (Hashmap)
  • Circular Singly Linked List
  • Doubly Linked List
  • Circular Doubly Linked List
  • Generalized Linked List
  • Heap
  • Binary Tree (Generation using 2 traversals left)
  • Binary Search Tree (Done recursively, TODO delete for all 3 cases)
  • AVL Tree
  • RB Tree

Algorithms implemented

  • Linear Search in Vector
  • Linear Search in SLL
  • Binary Search
  • Bubble Sort
  • Selection Sort
  • Insertion Sort
  • Merge Sort
  • Quick sort

Pros

  1. Relatively easy use of generic data structures and algorithms while being type-safe.
  2. Is cool, I mean what is cooler than preprocessor hijinks.

Cons

  1. This method is optimised for ease of use (probably for beginners/learners because it primarily works with pointers to structs), however, defining multiple data structures in the same file may (read WILL) lead to slower compilation times due to large compilation units because of the coupling of interfaces (headers & prototypes) and the actual functionality (function definitions) itself.
  2. Poor support (slow) for Intellisense and similar code intelligence and autocompletion tools.
  3. As all the functions are defined in the macro, with everything being in C, there is no way to apply access modifiers to helper functions for a cleaner interface and hacks like using double underscores seem inelegant.

This project is a side-effect of trying to develop a command-line tool for easier use of generic data structures for C, which splits the interface and the function definitions, leading to better compilation times.

Some helpful projects which helped me develop this

  1. Thought this was an original idea, but this repository I found later implemented this already - https://github.com/mystborn/GenericDataStructures

  2. I got the idea for the need of type-safe data structures by seeing this repository, otherwise it is perfect, especially the error handling is chef's kiss - https://github.com/MostafaTwfiq/C-DataStructures-And-Algorithms

About

Header-only, type-safe, generic data structures and algorithms for C.

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published