| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280 | <html lang="en"><head><title>Two dimensional arrays - avram - a virtual machine code interpreter</title><meta http-equiv="Content-Type" content="text/html"><meta name="description" content="avram - a virtual machine code interpreter"><meta name="generator" content="makeinfo 4.13"><link title="Top" rel="start" href="index.html#Top"><link rel="up" href="Type-Conversions.html#Type-Conversions" title="Type Conversions"><link rel="prev" href="One-dimensional-arrays.html#One-dimensional-arrays" title="One dimensional arrays"><link rel="next" href="Related-utility-functions.html#Related-utility-functions" title="Related utility functions"><link href="http://www.gnu.org/software/texinfo/" rel="generator-home" title="Texinfo Homepage"><meta http-equiv="Content-Style-Type" content="text/css"><style type="text/css"><!--  pre.display { font-family:inherit }  pre.format  { font-family:inherit }  pre.smalldisplay { font-family:inherit; font-size:smaller }  pre.smallformat  { font-family:inherit; font-size:smaller }  pre.smallexample { font-size:smaller }  pre.smalllisp    { font-size:smaller }  span.sc    { font-variant:small-caps }  span.roman { font-family:serif; font-weight:normal; }   span.sansserif { font-family:sans-serif; font-weight:normal; } --></style></head><body><div class="node"><a name="Two-dimensional-arrays"></a><p>Next: <a rel="next" accesskey="n" href="Related-utility-functions.html#Related-utility-functions">Related utility functions</a>,Previous: <a rel="previous" accesskey="p" href="One-dimensional-arrays.html#One-dimensional-arrays">One dimensional arrays</a>,Up: <a rel="up" accesskey="u" href="Type-Conversions.html#Type-Conversions">Type Conversions</a><hr></div><h5 class="subsubsection">3.1.4.3 Two dimensional arrays</h5><p>Several other functions in <samp><span class="file">matcon.h</span></samp> are meant to supportconversions between matrices represented as lists of lists and arraysin a variety of representations. Dense matrices either square or<a name="index-matrices-450"></a>rectangular are accommodated, and symmetric square matrices can bestored with redundant entries omitted in either upper trangular orlower triangular format.   <p>Similarly to the vector operations (<a href="One-dimensional-arrays.html#One-dimensional-arrays">One dimensional arrays</a>)these functions are intended to allow a developer to present aninterface to external libraries based on lists rather than arrays.   <p>The preferred convention for virtual code applications is to representa matrix as a list of lists of entities (typically numbers), withone list for each row of the matrix. For example, a 3 by 3 matrixcontaining a value of <code>aij</code> in the <code>i</code>-th row and the<code>j</code>-th column would be represented by this list of three lists.<pre class="example">     <        <a11,a12,a13>,        <a21,a22,a23>,        <a31,a32,a33>></pre>   <p class="noindent">Such a representation is convenient for manipulation byvirtual machine combinators, for example <code>transpose</code>(<a href="Transpose.html#Transpose">Transpose</a>), and is readily identified with the matrixit represents.   <p>If a matrix is symmetric (that is, with <code>aij</code> equal to<code>aji</code> for all values of <code>i</code> and <code>j</code>), only the lowertriangular portion needs to be stored because the other entries are<a name="index-triangular-matrix-451"></a>redundant.  The list representatation would be something like this.<pre class="example">     <        <a11>,        <a21,a22>,        <a31,a32,a33>></pre>   <p>Another alternative for representing a symmetric matrix is to store only theupper triangular portion. In this case, a list such as the following would be used.<pre class="example">     <        <a11,a12,a13>,        <a22,a23>,        <a33>></pre>   <p class="noindent">The upper and lower triangular representations are distinguishable bywhether or not the row lengths form an increasing sequence.   <p>In addition to representing symmetric matrices, these upper and lower<a name="index-symmetric-matrix-452"></a>triangular forms are also appropriate for representing matrices whoseremaining entries are zero, such as the factors in an LUdecomposition. <a name="index-LU-decomposition-453"></a><div class="defun">— Function: void <b>*avm_matrix_of_list</b> (<var>int square, int upper_triangular, int lower_triangular, int column_major, list operand, size_t item_size, list *message, int *fault</var>)<var><a name="index-g_t_002aavm_005fmatrix_005fof_005flist-454"></a></var><br><blockquote>        <p>This function converts a matrix in one of the list representationsabove to a contiguous array according to the given specifications. The array can contain elements of any fixed sized type of size<var>item_size</var>. The memory for it is allocated by this function andit should be freed by the caller when no longer needed.        <p>The input matrix is given by the list parameter, <var>operand</var>, andits format is described by the integer parameters <var>square</var>,<var>upper_triangular</var>, and <var>lower_triangular</var>. The number ofbytes occupied by each entry is given by <var>item_size</var>.        <p>To the extent these specifications are redundant, they are used forvalidation. If any of the following conditions is not met, the integerreferenced by <var>fault</var> is assigned a non-zero value and a copyof the message <code><'bad matrix specification'></code> represented asa list is assigned to the list referenced by <var>message</var>. Errorsare also possible due to insufficient memory.          <ul><li>The <var>operand</var> must be a list of lists of lists such thateach item of each item is has a length of <var>item_size</var>,and its items consist of character representations asrequired by <code>avm_value_of_list</code> (<a href="Primitive-types.html#Primitive-types">Primitive types</a>). <li>If the lengths of the top level lists in the <var>operand</var> form anincreasing sequence, the lower triangular representation is assumedand the <var>lower_triangular</var> parameter must have a non-zero value. <li>If the lengths of the top level lists in the <var>operand</var> form adecreasing sequence, the upper triangular representation is assumedand the <var>upper_triangular</var> parameter must have a non-zero value. <li>At least one of <var>upper_triangular</var> or <var>lower_triangular</var> mustbe zero. <li>If <var>square</var> has a non-zero value, then either all items of the<var>operand</var> must have the same length as the operand, or if it'striangular, then the longest one must have the same length as theoperand. <li>If the <var>operand</var> is neither square nor a triangular form, allitems of it are required to have the same length. </ul>        <p>The parameters <var>upper_triangular</var> or <var>lower_triangular</var> may beset to non-zero values even if the <var>operand</var> is not in one of theupper or lower triangular forms discussed above. In this case, the<var>operand</var> must be square or rectangular (i.e., with all items thesame length), and the following interpretations apply.          <ul><li>If <var>upper_triangular</var> is non-zero, the diagonal elements and theupper triangular portion of the input matrix are copied to the output. The lower triangle of the input is ignored and the lower triangle ofthe output is left uninitialized. <li>If <var>lower_triangular</var> is non-zero, the diagonal elements and thelower triangular portion of the input matrix are copied to the output. The upper triangle of the input is ignored and the upper triangle ofthe output is left uninitialized. </ul>        <p>The <var>column_major</var> parameter affects the form of the output array. If it is zero, then each row of the input matrix is stored in acontiguous block of memory in the output array, and if it is non-zero,each column is stored contiguously. <a name="index-Fortran-455"></a>The latter representation is alsoknown as Fortran order and may be required by library functionswritten in Fortran.        <p>In all cases when a triangular form is specified, part of the outputmatrix is left uninitialized. The redundant entries may be assigned ifrequired by the <code>avm_reflect_matrix</code> function (<a href="Related-utility-functions.html#Related-utility-functions">Related utility functions</a>). </p></blockquote></div><div class="defun">— Function: list <b>avm_list_of_matrix</b> (<var>void *matrix, int rows, int cols, size_t item_size, int *fault</var>)<var><a name="index-avm_005flist_005fof_005fmatrix-456"></a></var><br><blockquote><p>This function performs an inverse operation to<code>avm_matrix_of_list</code> by taking the address of a matrix stored asa contiguous array in the parameter <var>matrix</var> and constructing thelist representation as discussed above. Only square and rectangularmatrices in row major order are supported, but see<code>avm_matrix_transposition</code> for a way to convert between row major<a name="index-column-major-order-457"></a>and column major order (<a href="Related-utility-functions.html#Related-utility-functions">Related utility functions</a>).        <p>The parameters <var>rows</var>, <var>cols</var>, and <var>item_size</var> describethe form of the matrix. The list returned as a result will have alength of <var>rows</var>, and each item will be a list of length<var>cols</var>. Each item of the result corresponds to a row of thematrix, and each item of the items represents the an entry of thematrix as a list of length <var>item_size</var>. These items could bepassed to <code>avm_value_of_list</code>, for example, to obtain theirvalues (<a href="Primitive-types.html#Primitive-types">Primitive types</a>).        <p>Memory is allocated by this function to create the list, which can bereclaimed by <code>avm_dispose</code> (<a href="Simple-Operations.html#Simple-Operations">Simple Operations</a>). If there isinsufficient memory, the integer referenced by <var>fault</var> is assigneda non-zero value and the result returned is a list representation ofthe message <code><'memory overflow'></code>. The error message be reclaimedby the caller as well using <code>avm_dispose</code>. </p></blockquote></div>   <p>A packed storage representation for symmetric square matrices and<a name="index-packed-arrays-458"></a>triangular matrices is of interest because it is used by some libraryfunctions, notably those in <code>LAPACK</code>, to save memory and therebyaccommodate larger problems. In this representation, column major<a name="index-column-major-order-459"></a>order is assumed, and either the lower or the upper triangle of thematrix is not explicitly stored. For example, a lower triangular<a name="index-triangular-matrix-460"></a>matrix whose list representation corresponds to<pre class="example">     <        <a11>,        <a21,a22>,        <a31,a32,a33>,        <a41,a42,a43,a44>></pre>   <p class="noindent">would be stored according to the memory map<a name="index-matrix-memory-map-461"></a><pre class="example">     [a11 a21 a31 a41 a22 a32 a42 a33 a43 a44]</pre>   <p class="noindent">with <code>a11</code> at the beginning address. An upper triangular matrix<pre class="example">     <        <a11,a12,a13,a14>,        <a22,a23,a24>,        <a33,a34>,        <a44>></pre>   <p class="noindent">would be stored according to the memory map<pre class="example">     [a11 a12 a22 a13 a23 a33 a14 a24 a34 a44].</pre>   <p>A couple of functions converting between list representations andpacked array format are provided as described below.<div class="defun">— Function: void <b>*avm_packed_matrix_of_list</b> (<var>int upper_triangular, list operand, int n, size_t item_size, list *message, int *fault</var>)<var><a name="index-g_t_002aavm_005fpacked_005fmatrix_005fof_005flist-462"></a></var><br><blockquote>        <p>If the <var>operand</var> is a list in one of the triangular formsexplained above, then the <var>upper_triangular</var> parameter must beconsisitent with it, being non-zero if the <var>operand</var> is uppertriangular and zero otherwise.        <p>If the <var>operand</var> is not in a triangular form, then each item ofthe operand must be a list of length <var>n</var>. In this case, the<var>upper_triangular</var> parameter indicates which triangle of theoperand should be copied to the result, and the other triangle isignored.        <p>In either case, the operand must have a length of <var>n</var>, and theitems of its items must be lists of length <var>item_size</var> containingcharacter representations as required by <code>avm_value_of_list</code>(<a href="Primitive-types.html#Primitive-types">Primitive types</a>).        <p>If the input parameters are inconsistent or if there is insufficientmemory to allocate the result, the integer referenced by <var>fault</var>is assigned a non-zero value, and the list referenced by <var>message</var>is assigned a copy of the list representation of <code><'bad matrixspecification'></code> or <code><'memory overflow'></code>, respectively. Anon-empty message must be reclaimed by the caller using<code>avm_dispose</code> (<a href="Simple-Operations.html#Simple-Operations">Simple Operations</a>).        <p>If there are no errors, the result is a pointer to a packed arrayrepresentation of the <var>operand</var> as explained above. The memory forthis result is allocated by this function and should be freed by thecaller when no longer required. The number of bytes allocated will be<var>item_size</var> * (<var>n</var> * (<var>n</var> + 1))/2. </p></blockquote></div><div class="defun">— Function: list <b>avm_list_of_packed_matrix</b> (<var>int upper_trianguler,void *operand, int n, size_t item_size, int *fault</var>)<var><a name="index-avm_005flist_005fof_005fpacked_005fmatrix-463"></a></var><br><blockquote>        <p>This function performs an inverse operation to that of<code>avm_packed_matrix_of_list</code> given the address of a packed matrixstored according to one of the memory maps discussed above. The<var>operand</var> parameter holds the address, the parameter <var>n</var> givesthe number of rows, and the <var>upper_triangular</var> parameter specifieswhich of the two possible memory maps to assume.        <p>If there is sufficient memory, the result returned is a list in one ofthe triangular forms described above, being upper triangular if the<var>upper_triangular</var> parameter is non-zero, with values of length<var>item_size</var> taken from the array.        <p>In the event of a memory overflow, the integer referenced by<var>fault</var> is assigned a non-zero value and the result is a copy ofthe message <code><'memory overflow'></code> represented as a list. A<a name="index-segmentation-fault-464"></a>segmentation fault is possible if this function is passed an invalidpointer or dimension. </p></blockquote></div>   </body></html>
 |