1. Introduction
Geometric algebra (GA) is a powerful mathematical framework that unifies a variety of mathematical concepts, including complex numbers, quaternions, octonions, vectors, linear subspaces, orthogonal maps, and tensors, into a single uniform algebraic system. GA is capable of algebraically modeling many interesting types of geometry, including Euclidean, elliptic, hyperbolic, and conformal geometries. Research on practical applications of geometric algebra has expanded greatly in recent years. GA has applications in various fields such as computer graphics, robotics, classical mechanics, modern physics, electromagnetics, power systems, geographical information systems, and computer vision [
1,
2,
3,
4,
5].
As can be expected, there are many software libraries capable of utilizing GA for practical scientific computing and geometric modeling tasks. There are mainly three categories of GA software libraries: numerical and symbolic computational libraries, source code library generators, and optimizing subroutine generators. Computational GA libraries serve the purpose of performing general numeric or symbolic GA operations. GA library generators are designed to perform code generation for producing GA code libraries for practical use based on some specification of a selected GA. Finally, optimizing subroutine generators are capable of producing properly optimized code from GA multivector expressions and algorithms as a single block of subroutine code.
In this paper, the geometric algebra fulcrum library (GA-FuL) is introduced as a generic, wide-purpose software library for computing with geometric algebra. GA-FuL is capable of performing all three categories of other software libraries within a single, well-composed system. The high-level layered design of GA-FuL is based on a set of core intentions implemented through the principles of data-oriented programming. The ultimate goal of GA-FuL design is to produce a highly readable, maintainable, and extensible code base to properly serve the intended objectives of the library. In addition, the paper contains a brief exposition of some practical applications of GA-FuL, illustrating some of its capabilities.
GA-FuL is implemented in .NET Framework 8 using C# language. Accessing the library capabilities can be done through any .NET language, including, but not limited to, C#, F#, managed C++, and VisualBasic. The code generation capabilities of GA-FuL are generic, meaning any text-based coding language can be targeted using GA-FuL metaprogramming components.
The paper is organized into seven sections. After the introduction, 
Section 2 contains a brief mathematical overview of geometric algebra. 
Section 3 provides an exposition of some GA-based software libraries along with their main characteristics. 
Section 4 details the high-level design of GA-FuL, including core design intentions, data-oriented programming characteristics, and extensible layered design. 
Section 5 contains several illustrative and practical use cases of the library that highlight its potential for mathematical, scientific, and engineering applications. 
Section 6 provides some comparisons of GA-FuL with similar GA software libraries. Finally, the conclusions are provided in 
Section 7.
  2. Overview of Geometric Algebra
In this work, real scalars are denoted using regular small letters, matrices are denoted using regular capital letters, vectors are denoted using bold small letters, and multivectors are denoted using bold capital letters.
Assume a vector space 
 spanned by an ordered set of 
n orthonormal basis vectors 
, with the property
      
A geometric algebra 
 can be defined by extending the vector space 
 using a bilinear product on vectors called the geometric product, defined using the following:
The geometric product  maps a basis vector  to a real scalar  called the signature of the basis vector while constructing a new irreducible algebraic entity  from any two different basis vectors. The new entity  is called a 2-blade basis in the GA literature. Geometrically, a basis vector  could be understood to represent an axis-aligned 1-dimensional subspace (i.e., a coordinate axis) in . As a direct generalization, a 2-blade basis  could be understood to represent an axis-aligned 2-dimensional subspace (i.e., a coordinate plane) spanned by its two basis vectors  in . Additionally, the geometric product can operate on any ordered combination of basis vectors  with  and  to construct an algebraic representation of a k-dimensional subspace spanned by . This algebraic element is typically called a k-blade basis or a basis blade of grade k. By extension, a basis vector  in this system is called a basis 1-blade, and the unit real scalar  is the basis 0-blade. This kind of algebraic formulation is the foundation of one major expressive feature of GA: the representation and computation of subspaces of any dimension. Just as vectors in classical linear algebra represent 1-dimensional subspaces, GA blades represent k-dimensional subspaces. Moreover, common manipulations of subspaces such as projections and reflections are also applicable through blades, replacing matrices in classical linear algebra.
The number of unique 
k-blades basis in 
 is 
. By taking linear combinations of basis blades of the same grade 
k, a new 
-dimensional vector space 
 is obtained which has elements called 
k-vectors. This definition results in 
 and 
. Now, the real scalars and classical vectors are just elements in a more general 
-dimensional linear space 
, called the Grassmann space of multivectors. A multivector 
 is a linear combination of basis blades of all grades. A multivector 
 can be represented using one of the following equally useful forms: 
The first form (
2) is useful for representing multivectors as a linear list of scalars 
, ordered by the index 
i of their basis blades 
 in 
. For example, in a three-dimensional space, the multivector 
 can be expressed using a flat linear list of scalars: 
. The second form (3) is useful for expressing a multivector 
 using its 
k-vector parts 
. The linear operator 
, called the “grade extraction operator”, extracts the terms of 
 that exclusively belong to the 
k-vector space 
. The scalars of multivector 
 can be grouped by the grades of their corresponding basis blades into a graded list 
. The index-mapping function 
 in (4) maps a graded index 
 into the corresponding linear index 
i, where 
k is the grade and 
j is the index of the basis blade in 
, respectively.
The geometric product of two basis blades 
 is always another basis blade 
 up to a sign 
. A Cayley table of size 
 for the geometric product of basis blades could be constructed using (
1). Using the bilinear property of the geometric product for any two multivectors, their geometric product is another multivector in the same GA:
As a concrete example, assume the four-dimensional Euclidean vector space 
 with basis vectors 
 where 
 and 
 for 
. In this case, the Grassmann space 
 has five 
k-vector spaces with the following basis 
k-blades for each:
The full GA has 16 basis blades and any multivector 
 can be expressed in the form (3) using the index-mapping function 
 shown in 
Table 1.
The geometric product of any two basis blades is simple to compute using (
1). For example, 
. Relation (3) can be used to find the geometric product of any two multivectors.
Due to limited space, only a very small sample of geometric representations and operations made possible by the algebraic structure of GA is mentioned here. The interested reader can find much valuable information in the GA literature [
6,
7,
8,
9,
10,
11,
12,
13,
14,
15,
16,
17,
18,
19,
20,
21,
22,
23,
24]. 
Table 2 shows some basic multivector operations that are typically useful in practice. Please note that the squared norm is a real number that can be positive, zero, or negative. A multivector with a zero squared norm is called a null multivector.
Some useful bilinear products can be defined using the geometric product, such as the scalar, outer, and left-contraction products, as described in 
Table 3.
The outer product  constructs a k-blade  representing a k-dimensional subspace , spanned by a set of linearly independent vectors . The same blade, up to a scaling factor, could be constructed using any set of linearly independent vectors spanning . Thus, a blade in GA only encodes the general spatial attitude and dimension of the subspace it represents but not the details of any specific set of vectors spanning the subspace. The differentiating scalar factor of blades can hold additional information about the subspace, for example, its chirality (orientation), area\volume\hyper-volume, mass, etc.
The scalar product of k-blades  is one generalization of the inner product of vectors (which are 1-blades in GA). The scalar product is useful in computations that involve the relative orientations of two subspaces  of the same dimension in the space. For example, in Euclidean 3D space, the angle between two planes, represented by their two blades , is easily computed using the scalar product.
The left-contraction product is among the most useful generalizations of the inner product of vectors. The geometric operation represented by the left-contraction product of two blades  is first to project the subspace  on ; then, the orthogonal complement of the resulting subspace is taken, both in a single unified operation. In contrast to the outer product, which constructs larger subspaces from smaller ones, the left-contraction product takes a kind of difference of two subspaces. Similar interpretations can be associated with the right-contraction and inner products.
In GA, a nonnull blade can be used to represent a reflection operator or a projection operator, in addition to representing a subspace in the base vector space. Given a blade 
 of grade 
a, any other subspace blade 
 of grade 
b could be reflected or projected on the subspace that 
 represents using the following two grade-preserving sandwiching product expressions, respectively:
Using a single nonnull vector 
, it is possible to perform a reflection of an arbitrary blade 
 in the 
 dimensional hyperplane orthogonal to 
 using the grade-preserving sandwiching product expression 
. By repeating this operation with several nonnull vectors 
 and using the Cartan–Dieudonné theorem [
25], a simple and powerful algebraic method for representing all orthogonal transformations on subspaces within GA is attained using the versor product: 
, where 
, the geometric product of nonnull vectors, is called a versor.
When a versor 
 is of unit squared norm 
, it is called a rotor, which has many important applications in GA. More specifically, a rotor 
 can elegantly rotate any subspace 
 using a simple versor product 
, which is a natural generalization of unit complex numbers and unit quaternions in 2D and 3D, respectively. Additionally, simple rotors can be constructed in many ways, including the exponentiation of a bivector 
 using the following:
The direction of the bivector defines the plane of rotation, while the norm defines the amount of rotation, which is the usual Euclidean angle when .
Another important algebraic tool in GA is the outermorphism [
12]. An outermorphism 
 is a direct extension of a linear map 
 on vectors that preserves the outer product. For any multivectors 
 and scalar 
a, an outermorphism 
 satisfies the following:
Similarly, as 
 can be expressed using a matrix on the basis spanning 
, 
 can be expressed as a set of 
n matrices, with one matrix per 
k-vector basis spanning 
 [
7,
26,
27]. Many important geometric operations can be represented as outermorphisms, including linear projections, reflections, and versor products, and then invariantly applied to subspaces represented as blades, without the explicit need to express a subspace using any particular set of basis vectors that span the subspace [
7,
12].
By varying the signatures of GA’s basis vectors, it is possible to represent many nonlinear geometric objects using linear subspaces of the GA. For example, the widely used five-dimensional conformal geometric algebra (CGA) [
7] has five basis vectors, 
, with signatures 
, respectively. CGA is capable of representing Euclidean spheres, circles, point pairs, points, lines, and planes with blades defined using the five basis vectors. These CGA blades can be used as geometric objects and as geometric operators for reflection and projection. Additionally, CGA versors can be used as orthogonal transformation operators to encode conformal maps, including Euclidean transformations and uniform scaling. Projective geometric algebra (PGA) [
9,
11] is another important GA with many practical applications and highly interesting theoretical implications for geometric modeling. Vector geometric algebra (VGA) and homogeneous geometric algebra (HGA) [
7] are two simpler GAs with the same Euclidean basis vector signature 
 but different geometric interpretation for the basis blades. Additionally, a few other GAs exist that can model conic sections and surfaces in two- and three-dimensional Euclidean spaces along with their intersections and transformations [
23,
28,
29,
30,
31,
32,
33].
  4. The Geometric Algebra Fulcrum Library (GA-FuL)
GA-FuL (
https://github.com/ga-explorer/GeometricAlgebraFulcrumLib accessed on 15 June 2024) is an extensive .NET framework software library written in the C# programming language [
51], capable of performing computational tasks of all three categories of GA software systems. The name 
“Geometric Algebra Fulcrum” signifies two aspects of the system: mathematical and computational. For the mathematical aspect, geometric algebra is a unifying algebraic language that incorporates and generalizes many seemingly diverse algebraic tools under the same framework of mathematical operations [
12]. Such algebraic tools include, to name a few, real numbers, complex numbers, quaternions, octonions, and orthogonal matrices. From the experience of many researchers who have worked with GA, its most important feature is, however, its ability to unify the geometric reasoning process among many seemingly diverse fields of application domains [
52]. This effectively allows for easier cross-discipline communication and idea exchange on a fundamentally algebraic level. Thus, GA acts as a 
fulcrum for geometric reasoning across diverse scientific and engineering domains.
For the computational aspect, there is a pressing need for software tools and code libraries that act as pivot points, i.e., fulcrums, for developing several classes of computational systems based on multivectors and GA operations. Such computational systems include numerical prototyping, symbolic mathematical investigations, and optimized code generation, among others. Writing and maintaining a separate code base for each class of systems is impractical. GA-FuL is intended to play the role of a computational fulcrum for prototyping, geometric modeling, investigating relevant algorithms, and developing computational software libraries based on the geometric reasoning fulcrum of geometric algebra. Additionally, GA-FuL provides a complete application programming interface (API) for handling a wide spectrum of application domains.
  4.1. Core Design Intentions
The high-level design of GA-FuL targets a specific set of core design intentions (CDIs). The set of CDIs is a direct result of the experience gained during the development of the predecessor system, GMac. An overview of the CDIs is as follows:
- CDI-1: Abstracting multivector operations from concrete scalar representations.-  In practical computational applications, there are diversely useful representations for the mathematical concept of scalars, as classically introduced in abstract algebra [ 53- ]. The most common representation for numerical applications is for real scalars using the IEEE 754 floating-point number format [ 54- ]. For modern data-driven and machine learning applications [ 55- ], multi-dimensional arrays and tensors are the basic representations for scalar data [ 56- ]. Another very useful representation for scalars, used in most Computer Algebra Systems (CAS), is an expression tree which represents a symbolic scalar expression for mathematical manipulation applications [ 57- ]. The foundational requirement for GA-FuL design is to provide a unified generic implementation for storing and manipulating multivectors and performing common GA operations on any kind of useful scalar representation. 
 
- CDI-2: Reducing memory requirements for sparse multivectors.-  One major obstacle in the way of using GA for practical computations is the memory storage requirements imposed by the structure of multivectors. Storing a full multivector in a  n- -dimensional GA requires  -  scalars, while storing a full  k- -vector requires  -  scalars. For the simplest of numerical scalar representations, 32-bit floating point numbers, a single multivector in a 30-dimensional GA requires 8 GBytes of memory, while a full 15-vector one requires nearly 1.16 GBytes of memory, assuming the use of memory arrays to store the scalars. For most practical GA applications, however, there is rarely a need for storing full multivectors or full  k- -vectors, and only sparse multivectors are sufficient. One feature of GA is to enable the creation of linear models of nonlinear geometric objects by embedding the original space into a higher-dimension GA space. This typically results in multivectors representing geometric objects using a significantly reduced number of scalars, not a full-sized multivector or  k- -vector. For example, a sparse multivector in five-dimensional conformal GA [ 24- ], containing only 5 out of 32 scalars, can represent points, planes, and spheres in three-dimensional Euclidean space. As such, another important requirement for the GA-FuL design is to provide a set of generic, memory-efficient data structures for storing the scalars of sparse multivectors in high-dimensional GAs. 
 
- CDI-3: Providing metaprogramming capabilities.-  Generative programming [ 58- , 59- ] in general and metaprogramming [ 60- , 61- ] in particular incorporate the process of creating software systems that treat programs as data, enabling the transformation of existing programs or for the generation of new ones. This was the target of the predecessor system GMac for generating optimized computational numerical code from a series of GA expressions [ 47- , 49- ]. This design goal is carried on to GA-FuL, which would enable code generation targeting high-performance platforms such as CUDA, in addition to classical general purpose programming languages such as C\C++, C#, Java, JavaScript, Python, MATLAB scripts, etc. 
 
- CDI-4: Introducing a layered system design for a wide spectrum of uses. The complexity imposed by the previous CDIs must be organized and managed through a layered design of the system. Each layer should specialize in one aspect of the system such as storage management, processing, algebraic and geometric abstractions, etc. In addition, a system of such capabilities would have a wide range of users and use cases. Typically, a user would use the system to create a numerical\symbolic prototype for some geometric modeling ideas, and then after some experimentation, would use metaprogramming system capabilities to generate optimized code for the final model targeting a specific programming language or environment. The design of GA-FuL attempts to realize this layered approach to allow users of different backgrounds to select the suitable level of coding they can handle. The coding level ranges from the very high level of coordinate-independent prototyping using abstract GA operations up to a fully controlled low-level direct manipulation of scalars and coordinates for high-performance computing purposes. 
- CDI-5: Providing a unified, generic, and extensible API for several classes of applications. The final design goal of GA-FuL is to expose the system functionality through a good API. The API should have a unified public interface with uniform conventions to aid usability. Additionally, the API should be generic regarding the kinds of scalars and GAs it can handle, reflecting the capabilities of the underlying system. API extensibility is also important for future development of the system to aid in the addition of more features and widening system usage. Finally, the API should support the development of various classes of applications including, but not limited to, numerical prototyping computations, symbolic mathematical manipulations, signal processing, visualization, and metaprogramming. 
  4.2. Design Overview
During the initial design of GA-FuL, satisfying the set of CDIs using traditional object-oriented programming (OOP) was found to be non-practical. This is mainly due to the tendency of classical OOP practices to increase the code-base complexity of large systems such as GA-FuL. In this context, complexity specifically means the deep coupling of data and behavior code typically imposed by classical OOP principles, especially encapsulation and inheritance. This typically results in complicated relations between system classes and complex inheritance hierarchies, which lead to difficulties in understanding the design of large systems. If not properly mitigated, this can eventually result in reduced code understanding and difficulty in system maintenance and extensibility.
The solution found to be most useful was to use a newly emerging software design paradigm that while being compatible with OOP, also tends to produce a more readable, maintainable, and extensible code-base. The use of data-oriented programming (DOP) principles [
62], as the highest-level design paradigm, proved to be highly beneficial to many aspects of GA-FuL system design. The four core principles of DOP as expressed in [
62] are follow:
- DOP-1: Separating behavior code from data. This is a design tenet that advocates for a distinct division between behavior code and data. Following this DOP principle in OOP entails grouping the behavior code into methods for a static class. 
- DOP-2: Representing data with generic data structures. DOP is not dogmatic about the programming constructs used to employ and organize the code. Arrays\lists and dictionaries\maps are the two most widely used generic data structures in practice. However, one can also utilize other general data structures, such queues, trees, and sets. 
- DOP-3: Making data immutable. In DOP, due to isolation of representational data structures from behavior code, data mutation is not permitted. Instead, data modifications are carried out by generating new data structure versions. A variable’s reference can be updated to point to a different version of the data, but the actual value of the data must never change. 
- DOP-4: Separating data representation from data schema. Now that data and code are decoupled and generic immutable data structures are employed to describe it, the challenge is to articulate the shape of the data. The intended shape in DOP is represented by a data schema that is stored apart from the actual data. DOP-4’s primary advantage is that it gives developers the freedom to choose which data elements should have a schema and which ones should not. 
In GA-FuL, DOP-1 is implemented using thin wrapper classes around generic data structures holding the actual data. Extension methods  [
51] in static utility classes operate on the thin-wrapper classes to perform the desired behaviors. As for DOP-2, sparse algebraic objects, such as 
k-vectors and multivectors, are stored in dictionaries, while dense algebraic objects, such as matrices and multidimensional scalar arrays, are stored in classical array data structures. DOP-3 is accomplished through specialized classes called composers. A composer for a multivector, for example, performs a data transformation\construction transaction that, when completed properly, generates a valid dictionary containing valid data values that a multivector code wrapper class and extension methods can query and manipulate later. Finally, the DOP-4 principle is accomplished through the use of generic interfaces and abstract base classes, where the wrapper classes and extension methods manipulate data with a given generic interface or abstract class regardless of the actual data structure implementing the interface\class at any moment during program execution.
As a specific example of how the DOP principles in GA-FuL are implemented, the interface IIndexSet is used as a data schema to represent all kinds of index sets for basis blades (according to DOP-4). For representing a GA basis blade , concrete class implementations of this interface internally use a sorted set of non-negative integers , completely independent of any specific GA metric. There are specialized immutable classes implementing the IIndexSet interface for the empty index set; a single-element index set, a more efficient index set with largest index less than 64 (internally using a 64-bit unsigned integer); a dense index set of arbitrary size (using an array of integers); and a sparse index set of arbitrary size (internally using a hash-set object for storing the indices) (according to DOP-2,3). The class XGaBasisBlade is a thin wrapper around an IIndexSet object with member and extension methods for performing basic operations on basis blades such as the geometric and other bilinear products, the reverse operation, etc. (in accordance with DOP-1).
Another example is the generic interface IReadOnlyDictionary<IIndexSet, T> that is the main data schema (DOP-4) for storing a sparse list of (basis blade, scalar value) pairs for all kinds of multivectors in GA-FuL. There is a specialized immutable class (DOP-2,3) implementing this interface for zero multivectors, another for storing a single (basis blade, scalar value) pair, and one for an arbitrary sparse list of (basis blade, scalar value) pairs. The internal data of a new multivector can be constructed using the XGaMultivectorComposer<T> composer class (DOP-3) acting as a construction transaction management class (DOP-1). The composer class automatically selects the most efficient concrete data structure class implementing the IReadOnlyDictionary<IIndexSet, T> interface to be used as internal storage for the constructed multivector.
  4.3. GA-FuL Layers
CDI-4 is a core design intention that manages the complexity of GA-FuL components. 
Figure 1 illustrates the layered design of GA-FuL currently consisting of four layers, each having a set of specific roles and sub-layers.
  4.3.1. Algebra Layer
At the lowest level, the algebra layer is designed specifically to fulfill CDI-1 and CDI-2, in addition to the four DOP principles. Other layers in GA-FuL eventually utilize the functionalities provided by this layer. Components in the algebra layer mainly perform two functions:
- DOP-adhering representations for generic scalars, basis blades, multivectors, linear maps, etc. 
- DOP-adhering processing tasks on the representations. 
Real scalar representations are considered external to this layer. A scalar can be represented using any desired class or structure, including numeric and symbolic representations provided by external packages. The generic IScalarProcessor<T> interface represents a processor to perform basic operations on scalars of arbitrary type T. This is one form of DOP-3 adherence in GA-FuL design where a scalar processor transforms scalar data to fulfill desired operations. Such operations include, among others, basic arithmetic (negation, addition, subtraction, multiplication, division, and power), transcendental functions (trigonometric, exponential, logarithms, etc.), and zero equality testing. The derived interface INumericScalarProcessor<T> is useful for implementing concrete scalar processors on numerical types. In the current implementation, there are scalar processors for standard single\double precision floating-point real and complex numbers, arbitrary precision decimal\floating-point scalars, and arbitrary-precision rational numbers. In addition, there is a class implementing these operations on NumPy-like multi-dimensional arrays, and another for sampled signals for computational data-driven and signal processing applications. A second derived interface, ISymbolicScalarProcessor<T>, is useful for handling symbolic scalars typically used in a CAS. This includes a class capable of processing Wolfram Mathematica symbolic scalars represented by the provided Expr class. New implementations can be added at later time to augment GA-FuL with the ability to interact with other symbolic processing systems such as Maple, the MATLAB symbolic toolbox, Python’s SymPy package, etc. There is also a generic thin-wrapper class Scalar<T> composed over a scalar processor of type IScalarProcessor<T> and a scalar value of type T . This class is meant to make the GA-FuL API easier to use. Using this class, instead of the complicated scalar processor call w = scalarProcessor.Add(x, scalarProcessor.Times(y,z)), the user can simply write w = x + y * z. A similar DOP-adhering design is used for storing and manipulating most mathematical object representations in GA-FuL, including multivectors, the core GA mathematical object.
For representing GA’s multivector basis blades , this layer internally uses a sorted index set , completely independent of any specific GA metric. As illustrated in the previous section, the interface IIndexSet is used to represent such index sets. Basic operations on individual and pairs of basis blades, such as the reverse operation or geometric products, for example, are performed at the lowest level through specialized integer manipulation subroutines. In the current implementation, blades with arbitrary dimensions can be represented using dynamic list-based index sets, while basis blades with dimensions less than 64 can be represented using fixed-length 64-bit integers, where a 1 indicates the presence of a basis vector in the index set of the basis blade, and a 0 indicates its absence. Additionally, GAs with 12 dimensions or less use various lookup tables to accelerate operations on lower-dimensional basis, blades. This structure enables more efficient processing of low-dimensional basis blades while allowing for the handling of arbitrary high-dimensional ones if the application requires.
On the processing side, the class XGaMetric is used for basic processing of basis blades with a specified metric signature such as Euclidean, projective, conformal, etc. The signature is specified using two numbers , the number of basis vectors that square to  respectively. All remaining vectors in a basis blade are assumed to square to 1. In this way, no fixed dimension is predefined for any particular metric computation on basis blades. As in the case of scalars, the thin-wrapper class XGaSignedBasisBlade is composed over a IIndexSet member, a XGaMetric member, and an integer sign member that can only take values . In this way, operations on basis blades can be easily performed using simple member and static extension methods on the XGaSignedBasisBlade class, instead of more complicated calls to methods of an XGaMetric object.
The data of a k-vector are stored in an immutable dictionary of (index set, generic scalar) key-value pairs of type IReadOnlyDictionary<IIndexSet, T>; with keys of type IIndexSet and scalar of generic type T. The number of indices per index set for all keys in the dictionary is constant and equal to k, the grade of the k-vector. The data of a multivector are stored in an immutable dictionary containing (grade k, k-vector) key-value pairs of type IReadOnlyDictionary<int, XGaKVector<T>>;, where a key holds a unique grade k, and the value is a k-vector part of the multivector. In this way, all linear and bilinear operations on multivectors are reduced to operations on k-vectors, which greatly simplifies the implementation. Additionally, this design enables a highly sparse and flexible representation of multivectors of all kinds in GA-FuL.
The generic 
XGaProcessor<T> class, derived from 
XGaMetric, is the root for all multivector processors in this layer. Most operations on multivectors, some of which are described in 
Section 2, are implemented using static extension methods taking a 
XGaProcessor<T> object as the main argument. The current version of GA-FuL allows for the representation and manipulation of GA spaces with any number of dimensions. All GA metrics are also possible based on the previous work detailed in  [
26]. In addition, there are specialized processor classes for Euclidean, conformal [
7], and projective [
8,
9,
11] GAs. Additionally, a small hierarchy of thin-wrapper classes, shown in 
Figure 2, is implemented to simplify the GA-FuL API, as in the case for scalars and basis blades. This scheme allows for the memory-efficient storage of both dense low-dimensional and sparse high-dimensional multivectors.
One downside of this generic scheme is the computational performance for some applications. For this reason, there is a similar class hierarchy, rooted in the RGaFloat64Multivector class, optimized specifically for sparse multivectors of standard floating point scalars and GA spaces with fewer than 64 dimensions. For even higher-performance applications, the use of code generation is possible using the metaprogramming layer in GA-FuL described below. This flexible design gives the user a wide set of implementation options for various application domains within a single software framework. Up to the best of the authors’ knowledge, no other single GA library provides a similar set of balanced choices simultaneously.
Additional classes for commonly useful GA transformations are also implemented. These include classes for general outermorphisms, general orthogonal operators (using versors in GA), general rotations (using GA rotors), and reflections\projections (using GA subspaces as reflection\projection operators), as illustrated in 
Figure 3.
In addition to real scalar algebra and geometric algebra, there are other kinds of algebraic representations implemented in the GA-FuL algebra layer. These include algebraic representations for complex numbers, quaternions, polynomials, linear algebra objects (planar angles, classical 2D\3D\4D\nD vectors, matrices, and general linear maps), and sampled signals for signal processing applications.
  4.3.2. Modeling Layer
The modeling layer mainly targets the fulfillment of CDI-5. in this layer, there are mostly thin wrappers around classes from the algebra layer, with specific member and extension methods suitable for the intended functionality of each class. The calculus sub-layer, still in the design stage, is intended to perform geometric calculus operations on multivectors as described in the GA literature [
12,
14,
21,
22,
52]. The visualization sub-layer is intended to visualize geometric objects using suitable 2D\3D computer graphics methods. Currently, it is possible to generate JavaScript code for the 3D visualization and animation engine Babylon.js (
https://www.babylonjs.com accessed on 15 June 2024) based on static and animated geometric objects from the geometry layer. Additionally, more sophisticated videos can be generated by combining individual image frames using the Selenium browser automation project Chrome WebDrive for.NET (
https://www.nuget.org/packages/Selenium.WebDriver.ChromeDriver accessed on 15 June 2024). Some illustrative examples are included online under the GA-FuL code repository (
https://github.com/ga-explorer/GeometricAlgebraFulcrumLib/tree/main/GeometricAlgebraFulcrumLib.Visualizations accessed on 15 June 2024).
The geometry sub-layer contains the highest level of specialized classes for particular geometries utilizing GA, such as Euclidean, projective, and conformal GAs. For  example, the conformal GA for describing 3D geometric objects can be used, as in the following code example:
- // The predefined scalar processor for 64-bits 
- // floating point numbers 
- var scalarProcessor = ScalarProcessorOfFloat64.Instance; 
-   
- Create the CGA space object based on the selected 
- kind of scalars 
- var cga = XGaConformalSpace5D<double>.Create(scalarProcessor); 
-   
- // Encode 4 points as CGA null vectors 
- var a = cga.EncodeIpnsRoundPoint(3.5, 4.3, 2.6); 
- var b = cga.EncodeIpnsRoundPoint(-2.1, 3.4, 5); 
- var c = cga.EncodeIpnsRoundPoint(7.4, -1.5, -4.5); 
- var d = cga.EncodeIpnsRoundPoint(3, -2, 5); 
-   
- // Use the outer product to define the OPNS blade, encoding a 
- sphere passing through points a, b, and c 
- var sphere = a.Op(b).Op(c).Op(d); 
-   
- // Encode a line passing through a point parallel to 
- a direction vector 
- var line = cga.EncodeOpnsFlatLine( 
- scalarProcessor.Vector3D(3.5, 4.3, 2.6), 
- scalarProcessor.Vector3D(1, 1, 1) 
- ); 
-   
- // Project line on sphere to get a circle 
- var circle = line.ProjectOpnsOn(sphere); 
-   
- // Decode the circle to separate its individual Euclidean 
- // geometric components 
- var circleComponents = circle.DecodeOpnsRound(); 
-   
- // Center, radius, direction bivector, and normal of circle: 
- var center = circleComponents.CenterToVector3D(); 
- var radius = circleComponents.RealRadius; 
- var bivector = circleComponents.DirectionToBivector3D(); 
- var normal = circleComponents.NormalDirectionToVector3D(); 
As illustrated in the code, after defining a 5D-CGA space using an instance of the class XGaConformalSpace5D<T>, the user can perform the following tasks:
- Encode a geometric object as a CGA blade\multivector. For example, the 5D CGA blades can represent the direction vectors, bivectors, points, point pairs, circles, spheres, lines, and planes of 3D Euclidean space. Additionally, CGA versors can encode all Euclidean and conformal maps such as rotations, translations, inversions, and reflections. 
- Perform basic GA multivector algebraic operations on the encoded multivectors in the CGA space. 
- Use simple member and extension methods to perform high-level geometric operations on the encoded multivectors. Examples include reflections, intersections, projections, translations, and rotations. 
- Decode a CGA blade\multivector into a set of simpler components. For example, a 5D CGA blade representing a circle can be decoded into the circle’s center, radius, direction bivector, and normal vector. 
In the current implementation of GA-FuL, the base class 
XGaConformalSpace<T> and its two derived classes 
XGaConformalSpace4D<T> , 
XGaConformalSpace5D<T> are capable of handling not only CGA of any dimension but also PGA through the implementation of the powerful exposition in [
63]. The main advantage of handling CGA and PGA within the same algebraic space is the ability to freely mix geometric object representations and their interactions within a single API.
  4.3.3. Metaprogramming Layer
The metaprogramming layer mainly targets CDI-3 and is the highest-level layer in GA-FuL. The main purpose of this layer is to generate optimized code in a selected target programming language (TPL), given a sequence of operations on multivectors and other algebraic objects in GA-FuL. In essence, the components of this layer construct an optimizing compiler and code generator that takes an expression tree having scalar parameters and constant numbers as leafs and standard operations on scalars as internal nodes. The expression tree is automatically constructed using GA-FuL algebraic and geometric modeling components and is then optimized and transformed into TPL code through the optimizing compiler and code-generator components.
This layer is useful for software engineers wanting to create specialized code, all or part of which is automatically generated from operations on algebraic objects, especially GA multivectors. The typical sequence for using the components of this layer consists of the following stages:
- Initialize a metaprogramming context object (MCO) by instancing the MetaContext class defined in the meta-context sub-layer. 
- Use the MCO to define algebraic objects acting as input parameters to the computational block. 
- Use algebraic operations, provided by GA-FuL algebra and modeling layers, to describe the intended algebraic steps. 
- Select the expected output variables of the computational block from the algebraic objects computed. 
- Set the TPL names of the input, intermediate, and output scalar variables to be used in the final generated code. 
- Use the MCO to optimize the computational block. 
- Initialize the intended code composer object (CCO) by instancing one of the classes defined in the code composers sub-layer. 
- Generate the final TPL optimized computational code using the CCO. 
In the meta-context sub-layer, a special kind of scalar, called a “meta-expression scalar”, is used. Essentially, a meta-expression scalar, 
Figure 4, is an expression tree similar to the ones typically used in computer algebra systems but with additional functionality for metaprogramming tasks. A computational block is constructed step by step by the user’s code while being stored and optimized automatically inside the metaprogramming context object. Conceptually, a computational block consists of a sequence of assignment statements to TPL variables defining the required computations at the lowest level of scalar components of the algebraic objects. The conversion from the high-level GA\algebraic operations used in step 3 into low-level scalar operations is performed automatically by the components of the meta context sub-layer and managed by the MCO itself. Some of the variables in the computational block can be assumed by the user as independent, externally defined input parameters, with no attached left-hand-side meta-expressions. Other scalar variables are mostly intermediate ones, except for a few that are selected by the user as output variables of the computational block. The MCO manages the entire computational code construction and optimization process. The MCO contains factory objects to add constant numbers and input parameters to the computational block. The MCO can perform the following optimizations on the computational code before the final code-generation step:
- Propagation of constant values in metaexpressions on the right-hand side. 
- Extraction of common subexpressions in right-hand-side metaexpressions into intermediate variables for reuse. 
- Optional simplification of metaexpressions on the right hand side using an external computer algebra system\library. 
- Pruning of intermediate variables having constant values or repeated right-hand-side metaexpressions or those not being used for computing an output variable. 
- Optional reduction of the number of intermediate variables required. 
- Optional reduction of computational steps in the code through a genetic programming algorithm [ 64- , 65- , 66- ] with a (4 + 1) evolutionary strategy. 
The purpose of the CCO is to convert the computational block assignment statements stored in the MCO into TPL code. The low-level metaexpressions of the computational block can only contain standard operations on scalars, such as negation, addition, subtraction, multiplication, division, power, trigonometric functions, exponential, and logarithms. Thus, the code composers sub-layer can be extended to provide code generation capabilities for almost any target programming language that supports such operations. The code composers sub-layer contains abstract classes for additional advanced code-generation tasks. The user can utilize classes of this sub-layer capable of template-based code generation for creating a wide range of general TPL code organizations, with or without using GA computations. These range from small text code with a single code file\module,  to a large code library with complicated folder and code file structure.
  4.3.4. System Utilities Layer
The utilities layer provides low-level services to the components in the other GA-FuL layers. The basic data structure sub-layer contains a set of data structures to aid in data storage and exchange in the system. For example, several classes that implement the generic IReadOnlyDictionary<TKey, TValue> interface are part of this sub-layer. The text\LaTeX utilities sub-layer provides core services for formatted text generation and LaTeX code composition extensively used by the system. The text-generation capabilities of this sub-layer are extensive. There are classes for composing formatted text, and parametric text templates and composers capable of creating full hierarchies of folders containing text files. The code-generation utilities sub-layer performs various low-level code-generation tasks used by the metaprogramming layer and other GA-FuL components. This includes components to represent and construct language-agnostic abstract syntax trees (ASTs) and code generators that can compose code based on the ASTs. Finally, the web graphics utilities sub-layer is used by the visualization sub-layer for generating suitable web-based code for rendering desired graphics from algebraic specifications in GA-FuL.