|
| 1 | +.. ############################################################################ |
| 2 | +.. File : doc/conventions.rst |
| 3 | +.. ############################################################################ |
| 4 | +
|
| 5 | +************ |
| 6 | +Conventions |
| 7 | +************ |
| 8 | + |
| 9 | +Since the C++ and Fortran binding code are generated by SWIG-Fortran, most |
| 10 | +conventions are based on its defaults and built-in wrappers as described in |
| 11 | +`the SWIG+Fortran manual page`. This section attempts to describe all |
| 12 | +conventions used by Flibcpp that may be ambiguous to either a C++ or Fortran |
| 13 | +user. Since Flibcpp's target audience is Fortran users, the library tries to |
| 14 | +follows existing Fortran conventions, and this document attempts to remain true |
| 15 | +to the official Fortran nomenclature as described in the ISO standards. |
| 16 | + |
| 17 | +.. _the SWIG+Fortran manual page: https://github.com/swig-fortran/swig/blob/master/Doc/Manual/src/Fortran.md |
| 18 | + |
| 19 | +Basic types |
| 20 | +============== |
| 21 | + |
| 22 | +The C++ standard library features many numeric classes and functions that take |
| 23 | +a template parameter corresponding to the type of a single element used by the |
| 24 | +class or algorithm. All such classes or functions that Flibcpp includes are |
| 25 | +templated on: |
| 26 | + |
| 27 | + - 32-bit integers (``integer(4)`` or ``integer(C_INT32_T)``), with a suffix of |
| 28 | + ``Int4`` where applicable; |
| 29 | + - 64-bit integers (``integer(8)`` or ``integer(C_INT64_T)``), with a suffix of |
| 30 | + ``Int8`` where applicable; and |
| 31 | + - 64-bit reals (``real(8)`` or ``real(C_DOUBLE)``), with a suffix of |
| 32 | + ``Real8`` where applicable. |
| 33 | + |
| 34 | +In general, most templated C++ functions use Fortran generic procedures to |
| 35 | +allow a single procedure name to operate on multiple types, and only the |
| 36 | +derived types (such as ``VectorInt4``) require the suffix. |
| 37 | + |
| 38 | +Error handling |
| 39 | +============== |
| 40 | + |
| 41 | +Some modules support error handling for checking input values. In all cases, |
| 42 | +the error status and a message can be accessed and cleared through the main |
| 43 | +``flc`` module:: |
| 44 | + |
| 45 | + use flc, only : ierr, get_serr |
| 46 | + |
| 47 | + ! <snip> |
| 48 | + if (ierr /= 0) then |
| 49 | + write(1,*) "Error", ierr, ":", get_serr() |
| 50 | + ! Reset the error flag to indicate that the error has been successfully |
| 51 | + ! handled. |
| 52 | + ierr = 0 |
| 53 | + fi |
| 54 | + |
| 55 | +Since errors do not immediately exit, it is up to the application code to check |
| 56 | +for and clear them. Failing to clear the error code may cause the application |
| 57 | +to exit on a subsequent Flibcpp procedure call. |
| 58 | + |
| 59 | +.. _conventions_indexing: |
| 60 | + |
| 61 | +Indexing |
| 62 | +======== |
| 63 | + |
| 64 | +C and C++ use the convention that 0 corresponds to the first element in an |
| 65 | +array: specifically, it indicates an *offset* of zero from the array's |
| 66 | +beginning. In Fortran, the convention is for the first element to have an index |
| 67 | +of 1, so Flibcpp has the same convention. |
| 68 | + |
| 69 | +This convention makes native array integration straightforward. For example, |
| 70 | +when the :ref:`modules_algorithm_binary_search` algorithm is used with any |
| 71 | +value ``val`` in the sorted array ``arr``, the following logical expression |
| 72 | +will be true:: |
| 73 | + |
| 74 | + arr(binary_search(arr, val)) == val |
| 75 | + |
| 76 | +An additional convention Flibcpp uses is for an invalid index to have a value |
| 77 | +of zero. Thus, the :ref:`modules_algorithm_binary_search` algorithm returns |
| 78 | +zero if asked to find an element that's not present. |
| 79 | + |
| 80 | +Derived type behavior |
| 81 | +===================== |
| 82 | + |
| 83 | +The derived types defined by Flibcpp are all "proxy" objects that operate on |
| 84 | +*pointers* to C++-owned objects. Some derived type values (i.e. class |
| 85 | +instances) will "own" the associated data, while some will merely reference |
| 86 | +data owned by other C++ objects. |
| 87 | + |
| 88 | +.. note:: Memory management in SWIG-Fortran-generated code is unintuitive and |
| 89 | + could change. If you have feedback, please contact the author. |
| 90 | + |
| 91 | +Construction |
| 92 | +------------ |
| 93 | + |
| 94 | +Although Fortran separates the processes of allocation and initialization, C++ |
| 95 | +combines them into the single act of *construction*. Thus the proxy objects in |
| 96 | +Flibcpp can either be in an unassigned, deallocated state or in an |
| 97 | +allocated and internally consistent state. |
| 98 | + |
| 99 | +Each derived type, such as the :ref:`modules_string_type`, is constructed using |
| 100 | +a *module procedure interface* with the same name as the type. This procedure, |
| 101 | +defined in the same module as the derived type, is a ``function`` that returns |
| 102 | +a "constructed" type:: |
| 103 | + |
| 104 | + use flc_string, only : String |
| 105 | + type(String) :: s |
| 106 | + |
| 107 | + s = String() |
| 108 | + |
| 109 | +Most classes have more than one constructor (i.e. procedure interface) that |
| 110 | +gives the object a more substantial initial state. For example, numeric vectors |
| 111 | +can be constructed directly from a Fortran array:: |
| 112 | + |
| 113 | + use flc_vector, only : Vector => VectorInt4 |
| 114 | + type(Vector) :: v, v2 |
| 115 | + v = Vector([1, 2, 3, 5, 8, 13]) |
| 116 | + |
| 117 | +Assignment |
| 118 | +---------- |
| 119 | + |
| 120 | +SWIG-Fortran, and consequently Flibcpp, defines assignment operators for its |
| 121 | +types that control memory ownership. Assignment for these types can be grouped |
| 122 | +into two categories: assignment directly from a Flibcpp function return value, |
| 123 | +and assignment from an existing Flibcpp derived type value. |
| 124 | + |
| 125 | +Flibcpp (or any other SWIG-Fortran) wrapper code sets the correct ownership |
| 126 | +flag on a return value: non-owning for raw pointers and references; owning for |
| 127 | +return-by-value. When the left-hand side of an assignment is uninitialized, it |
| 128 | +captures the returned value and obtains the correct ownership flag. If the |
| 129 | +left-hand side *is* initialized, it is automatically destroyed first. |
| 130 | + |
| 131 | +Assignment from within Fortran is like pointer assignment. The left-hand side |
| 132 | +becomes a non-owning reference to the right-hand side. |
| 133 | + |
| 134 | +Destruction |
| 135 | +----------- |
| 136 | + |
| 137 | +Unlike native allocatable Fortran types, Flibcpp derived types are not |
| 138 | +automatically deallocated when ending a procedure. Therefore to avoid |
| 139 | +leaking memory, these derived type values must be explicitly cleaned up and |
| 140 | + |
| 141 | +released. This is done by the type-bound subroutine named ``release``:: |
| 142 | + |
| 143 | + type(Vector), intent(inout) :: v |
| 144 | + call v%release() |
| 145 | + |
| 146 | +If the value ``v`` above owns the associated memory (i.e. if it was constructed |
| 147 | +in user code), then Flibcpp cleans up and deallocates the C++ instance, and |
| 148 | +sets ``v`` to an uninitialized state. If ``v`` merely points to existing C++ |
| 149 | +data, i.e. if it was assigned to the return result of a C++ accessor, then |
| 150 | +Flibcpp will simply set ``v`` to an uninitialized state. |
| 151 | + |
| 152 | +.. ############################################################################ |
| 153 | +.. end of doc/conventions.rst |
| 154 | +.. ############################################################################ |
0 commit comments