Skip to content

Commit be743af

Browse files
authored
Merge pull request #3 from swig-fortran/docs
Improve documentation
2 parents 4375f75 + 4ee5dd0 commit be743af

14 files changed

+299
-102
lines changed

doc/conventions.rst

+154
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,154 @@
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+
.. ############################################################################

doc/index.rst

+1
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ which wrap the C++ library.
3434
:caption: Contents
3535

3636
introduction.rst
37+
conventions.rst
3738
modules.rst
3839
examples.rst
3940
conclusion.rst

doc/modules.rst

-17
Original file line numberDiff line numberDiff line change
@@ -15,23 +15,6 @@ header, can be obtained via::
1515

1616
use flc_algorithm, only : sort
1717

18-
All templated routines are instantiated with 32- and 64-bit signed integers
19-
(``integer(4)`` and ``integer(8)``), and double-precision floats (``real(8)``).
20-
21-
Some modules support error handling for checking input values. In all cases,
22-
the error status and a message can be accessed and cleared through the main
23-
``flc`` module::
24-
25-
use flc, only : ierr, get_serr
26-
27-
! <snip>
28-
if (ierr /= 0) then
29-
write(1,*) "Error", ierr, ":", get_serr()
30-
! Reset the error flag to indicate that the error has been successfully
31-
! handled.
32-
ierr = 0
33-
fi
34-
3518
.. toctree::
3619
modules/algorithm.rst
3720
modules/chrono.rst

doc/modules/algorithm.rst

+4
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,8 @@ zero.
7777
Searching
7878
=========
7979

80+
.. _modules_algorithm_binary_search:
81+
8082
binary_search
8183
-------------
8284

@@ -144,6 +146,8 @@ Example::
144146

145147
call minmax_element(iarr, min_idx, max_idx) ! min_idx == 3, max_idx == 6
146148

149+
.. _modules_algorithm_set_operations:
150+
147151
Set operations
148152
==============
149153

doc/modules/set.rst

+3-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,9 @@
88
Set
99
***
1010

11-
Sorted sets with unions, intersections, etc.
11+
Sorted sets with unions, intersections, etc., are not yet implemented as
12+
objects. However, the :ref:`modules_algorithm_set_operations` for arrays can
13+
replicate some basic Set-like functionality.
1214

1315
.. ############################################################################
1416
.. end of doc/modules/set.rst

doc/modules/string.rst

+16-15
Original file line numberDiff line numberDiff line change
@@ -11,14 +11,16 @@ String
1111
The string module includes the ``String`` derived type and a handful of string
1212
conversion functions.
1313

14+
.. _modules_string_type:
15+
1416
String type
1517
===========
1618

1719
The C++ standard library "string" is a dynamically resizable, mutable character
1820
array.
1921

20-
Construction and destruction
21-
----------------------------
22+
Constructors
23+
------------
2224

2325
Strings are constructed using three interface functions:
2426

@@ -31,7 +33,7 @@ Strings are constructed using three interface functions:
3133
Here are three examples of initialization::
3234

3335
use flc_string, only : String
34-
type(String) :: v
36+
type(String) :: s
3537

3638
s = String()
3739
! s%size() == 0
@@ -40,6 +42,17 @@ Here are three examples of initialization::
4042
! s%get(i) == "!"
4143
s = String("I am a string!")
4244

45+
46+
Character element access
47+
------------------------
48+
49+
The number of characters in the string is returned by the bound function
50+
``size``. The ``get`` function returns the character at an index; and ``front``
51+
and ``back`` are aliases for ``get(1)`` and ``get(v%size())``, respectively.
52+
53+
.. important:: Unlike the C++ version of this class, **strings in Flibcpp
54+
use 1-offset indexing**. See :ref:`conventions_indexing`.
55+
4356
Modification
4457
------------
4558

@@ -61,18 +74,6 @@ specified index::
6174
call s%set(1, "8")
6275
call s%set(s%size(), "D")
6376

64-
.. important:: Unlike the C++ version of this class, **strings in Flibcpp
65-
use 1-offset indexing**. This means that ``s%get(1)`` is the same as the C++
66-
``s[0]``: it returns the first element (i.e. the element with an offset of
67-
zero).
68-
69-
Access
70-
------
71-
72-
The size of a string is returned by the bound function ``size``; ``get``
73-
returns the character at an index; and ``front`` and ``back`` are aliases for
74-
``get(1)`` and ``get(v%size())``, respectively.
75-
7677
Search
7778
------
7879

doc/modules/vector.rst

+2-2
Original file line numberDiff line numberDiff line change
@@ -83,8 +83,8 @@ The size of a vector is returned by the bound function ``size``; ``get``
8383
returns the value at an index; and ``front`` and ``back`` are aliases for
8484
``get(1)`` and ``get(v%size())``, respectively.
8585

86-
Additionally, ``front_ref``, ``back_ref``, and ``get_ref`` return pointers to
87-
the elements of the array.
86+
Additionally, ``front_ref``, ``back_ref``, and ``get_ref`` return Fortran
87+
pointers to the elements of the array.
8888

8989
.. warning:: Array element pointers are valid **only** as long as the vector's
9090
size is not changed. Calling ``erase``, ``push_back``, and so forth will

include/flc.i

+17-13
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,10 @@
1111
#error "To import the FLC module correctly, use ``%include \"import_flc.i\"``"
1212
#endif
1313

14+
/* -------------------------------------------------------------------------
15+
* Header definition macros
16+
* ------------------------------------------------------------------------- */
17+
1418
%define %flc_add_header
1519
%insert("fbegin") %{
1620
! Flibcpp project, https://github.com/swig-fortran/flibcpp
@@ -28,26 +32,27 @@
2832

2933
%flc_add_header
3034

31-
/************************
35+
/* -------------------------------------------------------------------------
3236
* Exception handling
33-
************************/
37+
* ------------------------------------------------------------------------- */
3438

35-
/* Rename the error variables' internal C symbols */
39+
// Rename the error variables' internal C symbols
3640
#define SWIG_FORTRAN_ERROR_INT flc_ierr
3741
#define SWIG_FORTRAN_ERROR_STR flc_get_serr
3842

39-
/* Restore names in the wrapper code */
43+
// Restore names in the wrapper code
4044
%rename(ierr) flc_ierr;
4145
%rename(get_serr) flc_get_serr;
4246

43-
/* Unless we're directly building this module, delay exception handling */
47+
// Unless we're directly building this module, delay exception handling
4448
#ifndef SWIGIMPORTED
4549
%include <exception.i>
4650
#endif
4751

48-
/******************************
52+
/* -------------------------------------------------------------------------
4953
* Data types and instantiation
50-
******************************/
54+
* ------------------------------------------------------------------------- */
55+
5156
%{
5257
#include <cstdint>
5358
using std::int32_t;
@@ -62,9 +67,9 @@ using std::size_t;
6267
%template(DST) SRC<double>;
6368
%enddef
6469

65-
/************************
70+
/* -------------------------------------------------------------------------
6671
* Array view translation
67-
************************/
72+
* ------------------------------------------------------------------------- */
6873

6974
%include <typemaps.i>
7075
%apply (SWIGTYPE *DATA, size_t SIZE) {
@@ -77,17 +82,16 @@ using std::size_t;
7782
(const int64_t *DATA, size_t DATASIZE),
7883
(const double *DATA, size_t DATASIZE) };
7984

80-
/************************
85+
/* -------------------------------------------------------------------------
8186
* Version information
82-
*
83-
* Linked into auto-generated file flibcpp_version.cpp
84-
************************/
87+
* ------------------------------------------------------------------------- */
8588

8689
%apply char* { const char flibcpp_version[] };
8790
%fortranbindc flibcpp_version_major;
8891
%fortranbindc flibcpp_version_minor;
8992
%fortranbindc flibcpp_version_patch;
9093

94+
// These symbols are defined in the CMake-generated `flibcpp_version.cpp`
9195
%inline %{
9296
extern "C" {
9397
extern const char flibcpp_version[];

0 commit comments

Comments
 (0)