Skip to content

Commit 8f0cd69

Browse files
authored
Merge branch 'fortran-lang:master' into activations
2 parents bc2bf5a + 8672fe1 commit 8f0cd69

17 files changed

+1314
-75
lines changed

config/fypp_deployment.py

+6-6
Original file line numberDiff line numberDiff line change
@@ -122,11 +122,11 @@ def fpm_build(args,unknown):
122122
flags= flags + unknown[idx+1]
123123
#==========================================
124124
# build with fpm
125-
subprocess.run(["fpm build"]+
126-
[" --compiler "]+[FPM_FC]+
127-
[" --c-compiler "]+[FPM_CC]+
128-
[" --cxx-compiler "]+[FPM_CXX]+
129-
[" --flag "]+[flags], shell=True, check=True)
125+
subprocess.run("fpm build"+
126+
" --compiler "+FPM_FC+
127+
" --c-compiler "+FPM_CC+
128+
" --cxx-compiler "+FPM_CXX+
129+
" --flag \"{}\"".format(flags), shell=True, check=True)
130130
return
131131

132132
if __name__ == "__main__":
@@ -137,7 +137,7 @@ def fpm_build(args,unknown):
137137
parser.add_argument("--vpatch", type=int, default=0, help="Project Version Patch")
138138

139139
parser.add_argument("--njob", type=int, default=4, help="Number of parallel jobs for preprocessing")
140-
parser.add_argument("--maxrank",type=int, default=7, help="Set the maximum allowed rank for arrays")
140+
parser.add_argument("--maxrank",type=int, default=4, help="Set the maximum allowed rank for arrays")
141141
parser.add_argument("--with_qp",action='store_true', help="Include WITH_QP in the command")
142142
parser.add_argument("--with_xdp",action='store_true', help="Include WITH_XDP in the command")
143143
parser.add_argument("--lnumbering",action='store_true', help="Add line numbering in preprocessed files")

doc/specs/stdlib_linalg.md

+107
Original file line numberDiff line numberDiff line change
@@ -1459,3 +1459,110 @@ If `err` is not present, exceptions trigger an `error stop`.
14591459
{!example/linalg/example_inverse_function.f90!}
14601460
```
14611461

1462+
## `get_norm` - Computes the vector norm of a generic-rank array.
1463+
1464+
### Status
1465+
1466+
Experimental
1467+
1468+
### Description
1469+
1470+
This `pure subroutine` interface computes one of several vector norms of `real` or `complex` array \( A \), depending on
1471+
the value of the `order` input argument. \( A \) may be an array of any rank.
1472+
1473+
Result `nrm` returns a `real`, scalar norm value for the whole array; if `dim` is specified, `nrm` is a rank n-1
1474+
array with the same shape as \(A \) and dimension `dim` dropped, containing all norms evaluated along `dim`.
1475+
1476+
### Syntax
1477+
1478+
`call ` [[stdlib_linalg(module):get_norm(interface)]] `(a, nrm, order, [, dim, err])`
1479+
1480+
### Arguments
1481+
1482+
`a`: Shall be a rank-n `real` or `complex` array containing the data. It is an `intent(in)` argument.
1483+
1484+
`nrm`: if `dim` is absent, shall be a scalar with the norm evaluated over all the elements of the array. Otherwise, an array of rank `n-1`, and a shape similar
1485+
to that of `a` with dimension `dim` dropped.
1486+
1487+
`order`: Shall be an `integer` value or a `character` flag that specifies the norm type, as follows. It is an `intent(in)` argument.
1488+
1489+
| Integer input | Character Input | Norm type |
1490+
|------------------|------------------|---------------------------------------------------------|
1491+
| `-huge(0)` | `'-inf', '-Inf'` | Minimum absolute value \( \min_i{ \left|a_i\right| } \) |
1492+
| `1` | `'1'` | 1-norm \( \sum_i{ \left|a_i\right| } \) |
1493+
| `2` | `'2'` | Euclidean norm \( \sqrt{\sum_i{ a_i^2 }} \) |
1494+
| `>=3` | `'3','4',...` | p-norm \( \left( \sum_i{ \left|a_i\right|^p }\right) ^{1/p} \) |
1495+
| `huge(0)` | `'inf', 'Inf'` | Maximum absolute value \( \max_i{ \left|a_i\right| } \) |
1496+
1497+
`dim` (optional): Shall be a scalar `integer` value with a value in the range from `1` to `n`, where `n` is the rank of the array. It is an `intent(in)` argument.
1498+
1499+
`err` (optional): Shall be a `type(linalg_state_type)` value. This is an `intent(out)` argument. If `err` is not present, the function is `pure`.
1500+
1501+
### Return value
1502+
1503+
By default, the return value `nrm` is a scalar, and contains the norm as evaluated over all elements of the generic-rank array \( A \).
1504+
If the optional `dim` argument is present, `nrm` is a rank `n-1` array with the same shape as \( A \) except
1505+
for dimension `dim`, that is collapsed. Each element of `nrm` contains the 1D norm of the elements of \( A \),
1506+
evaluated along dimension `dim` only.
1507+
1508+
Raises `LINALG_ERROR` if the requested norm type is invalid.
1509+
Raises `LINALG_VALUE_ERROR` if any of the arguments has an invalid size.
1510+
If `err` is not present, exceptions trigger an `error stop`.
1511+
1512+
### Example
1513+
1514+
```fortran
1515+
{!example/linalg/example_get_norm.f90!}
1516+
```
1517+
1518+
## `norm` - Computes the vector norm of a generic-rank array.
1519+
1520+
### Status
1521+
1522+
Experimental
1523+
1524+
### Description
1525+
1526+
This function computes one of several vector norms of `real` or `complex` array \( A \), depending on
1527+
the value of the `order` input argument. \( A \) may be an array of any rank.
1528+
1529+
### Syntax
1530+
1531+
`x = ` [[stdlib_linalg(module):norm(interface)]] `(a, order, [, dim, err])`
1532+
1533+
### Arguments
1534+
1535+
`a`: Shall be a rank-n `real` or `complex` array containing the data. It is an `intent(in)` argument.
1536+
1537+
`order`: Shall be an `integer` value or a `character` flag that specifies the norm type, as follows. It is an `intent(in)` argument.
1538+
1539+
| Integer input | Character Input | Norm type |
1540+
|------------------|------------------|---------------------------------------------------------|
1541+
| `-huge(0)` | `'-inf', '-Inf'` | Minimum absolute value \( \min_i{ \left|a_i\right| } \) |
1542+
| `1` | `'1'` | 1-norm \( \sum_i{ \left|a_i\right| } \) |
1543+
| `2` | `'2'` | Euclidean norm \( \sqrt{\sum_i{ a_i^2 }} \) |
1544+
| `>=3` | `'3','4',...` | p-norm \( \left( \sum_i{ \left|a_i\right|^p }\right) ^{1/p} \) |
1545+
| `huge(0)` | `'inf', 'Inf'` | Maximum absolute value \( \max_i{ \left|a_i\right| } \) |
1546+
1547+
`dim` (optional): Shall be a scalar `integer` value with a value in the range from `1` to `n`, where `n` is the rank of the array. It is an `intent(in)` argument.
1548+
1549+
`err` (optional): Shall be a `type(linalg_state_type)` value. This is an `intent(out)` argument. If `err` is not present, the function is `pure`.
1550+
1551+
### Return value
1552+
1553+
By default, the return value `x` is a scalar, and contains the norm as evaluated over all elements of the generic-rank array \( A \).
1554+
If the optional `dim` argument is present, `x` is a rank `n-1` array with the same shape as \( A \) except
1555+
for dimension `dim`, that is dropped. Each element of `x` contains the 1D norm of the elements of \( A \),
1556+
evaluated along dimension `dim` only.
1557+
1558+
Raises `LINALG_ERROR` if the requested norm type is invalid.
1559+
Raises `LINALG_VALUE_ERROR` if any of the arguments has an invalid size.
1560+
If `err` is not present, exceptions trigger an `error stop`.
1561+
1562+
### Example
1563+
1564+
```fortran
1565+
{!example/linalg/example_norm.f90!}
1566+
```
1567+
1568+

example/linalg/CMakeLists.txt

+2
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,8 @@ ADD_EXAMPLE(blas_gemv)
2828
ADD_EXAMPLE(lapack_getrf)
2929
ADD_EXAMPLE(lstsq1)
3030
ADD_EXAMPLE(lstsq2)
31+
ADD_EXAMPLE(norm)
32+
ADD_EXAMPLE(get_norm)
3133
ADD_EXAMPLE(solve1)
3234
ADD_EXAMPLE(solve2)
3335
ADD_EXAMPLE(solve3)

example/linalg/example_get_norm.f90

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
! Vector norm: demonstrate usage of the function interface
2+
program example_get_norm
3+
use stdlib_linalg, only: get_norm, linalg_state_type
4+
implicit none
5+
6+
real :: a(3,3), nrm, nrmd(3)
7+
integer :: j
8+
type(linalg_state_type) :: err
9+
10+
! a = [ -3.00000000 0.00000000 3.00000000
11+
! -2.00000000 1.00000000 4.00000000
12+
! -1.00000000 2.00000000 5.00000000 ]
13+
a = reshape([(j-4,j=1,9)], [3,3])
14+
15+
print "(' a = [ ',3(g0,3x),2(/9x,3(g0,3x)),']')", transpose(a)
16+
17+
! Norm with integer input
18+
call get_norm(a, nrm, 2)
19+
print *, 'Euclidean norm = ',nrm ! 8.30662346
20+
21+
! Norm with character input
22+
call get_norm(a, nrm, '2')
23+
print *, 'Euclidean norm = ',nrm ! 8.30662346
24+
25+
! Euclidean norm of row arrays, a(i,:)
26+
call get_norm(a, nrmd, 2, dim=2)
27+
print *, 'Rows norms = ',nrmd ! 4.24264050 4.58257580 5.47722578
28+
29+
! Euclidean norm of columns arrays, a(:,i)
30+
call get_norm(a, nrmd, 2, dim=1)
31+
print *, 'Columns norms = ',nrmd ! 3.74165750 2.23606801 7.07106781
32+
33+
! Infinity norms
34+
call get_norm(a, nrm, 'inf')
35+
print *, 'maxval(||a||) = ',nrm ! 5.00000000
36+
37+
call get_norm(a, nrmd, 'inf', dim=2)
38+
print *, 'maxval(||a(i,:)||) = ',nrmd ! 3.00000000 4.00000000 5.00000000
39+
40+
call get_norm(a, nrm, '-inf')
41+
print *, 'minval(||a||) = ',nrm ! 0.00000000
42+
43+
call get_norm(a, nrmd, '-inf', dim=1)
44+
print *, 'minval(||a(:,i)||) = ',nrmd ! 1.00000000 0.00000000 3.00000000
45+
46+
! Catch Error:
47+
! [norm] returned Value Error: dimension 4 is out of rank for shape(a)= [3, 3]
48+
call get_norm(a, nrmd, 'inf', dim=4, err=err)
49+
print *, 'invalid: ',err%print()
50+
51+
end program example_get_norm

example/linalg/example_norm.f90

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
! Vector norm: demonstrate usage of the function interface
2+
program example_norm
3+
use stdlib_linalg, only: norm, linalg_state_type
4+
implicit none
5+
6+
real :: a(3,3),na
7+
integer :: j
8+
type(linalg_state_type) :: err
9+
10+
! a = [ -3.00000000 0.00000000 3.00000000
11+
! -2.00000000 1.00000000 4.00000000
12+
! -1.00000000 2.00000000 5.00000000 ]
13+
a = reshape([(j-4,j=1,9)], [3,3])
14+
15+
print "(' a = [ ',3(g0,3x),2(/9x,3(g0,3x)),']')", transpose(a)
16+
17+
! Norm with integer input
18+
print *, 'Euclidean norm = ',norm(a, 2) ! 8.30662346
19+
20+
! Norm with character input
21+
print *, 'Euclidean norm = ',norm(a, '2') ! 8.30662346
22+
23+
! Euclidean norm of row arrays, a(i,:)
24+
print *, 'Rows norms = ',norm(a, 2, dim=2) ! 4.24264050 4.58257580 5.47722578
25+
26+
! Euclidean norm of columns arrays, a(:,i)
27+
print *, 'Columns norms = ',norm(a, 2, dim=1) ! 3.74165750 2.23606801 7.07106781
28+
29+
! Infinity norms
30+
print *, 'maxval(||a||) = ',norm(a, 'inf') ! 5.00000000
31+
print *, 'maxval(||a(i,:)||) = ',norm(a, 'inf', dim=2) ! 3.00000000 4.00000000 5.00000000
32+
print *, 'minval(||a||) = ',norm(a, '-inf') ! 0.00000000
33+
print *, 'minval(||a(:,i)||) = ',norm(a, '-inf', dim=1) ! 1.00000000 0.00000000 3.00000000
34+
35+
! Catch Error:
36+
! [norm] returned Value Error: dimension 4 is out of rank for shape(a)= [3, 3]
37+
print *, 'invalid: ',norm(a,'inf', dim=4, err=err)
38+
print *, 'error = ',err%print()
39+
40+
end program example_norm

include/common.fypp

+101
Original file line numberDiff line numberDiff line change
@@ -298,4 +298,105 @@ ${prefix + joinstr.join([line.strip() for line in txt.split("\n")]) + suffix}$
298298
#:endif
299299
#:enddef
300300

301+
#!
302+
#! Generates a list of loop variables
303+
#!
304+
#! Args:
305+
#! varname(str): Name of the variable to be used as prefix
306+
#! n (int): Number of loop variables to be created
307+
#! offset (int): Optional index offset
308+
#!
309+
#! Returns:
310+
#! Variable definition string
311+
#!
312+
#! E.g.,
313+
#! loop_variables('j', 5)
314+
#! -> "j1, j2, j3, j4, j5
315+
#!
316+
#:def loop_variables(varname, n, offset=0)
317+
#:assert n > 0
318+
#:call join_lines(joinstr=", ")
319+
#:for i in range(1, n + 1)
320+
${varname}$${i+offset}$
321+
#:endfor
322+
#:endcall
323+
#:enddef
324+
325+
#! Generates an array shape specifier from an N-D array size
326+
#!
327+
#! Args:
328+
#! name (str): Name of the original variable
329+
#! rank (int): Rank of the original variable
330+
#! offset(int): optional offset of the dimension loop (default = 0)
331+
#!
332+
#! Returns:
333+
#! Array rank suffix string enclosed in braces
334+
#!
335+
#! E.g.,
336+
#! shape_from_array_size('mat', 5)}$
337+
#! -> (size(mat,1),size(mat,2),size(mat,3),size(mat,4),size(mat,5))
338+
#! shape_from_array_size('mat', 5, 2)}$
339+
#! -> (size(mat,3),size(mat,4),size(mat,5),size(mat,6),size(mat,7))
340+
#!
341+
#:def shape_from_array_size(name, rank, offset=0)
342+
#:assert rank > 0
343+
#:call join_lines(joinstr=", ", prefix="(", suffix=")")
344+
#:for i in range(1, rank + 1)
345+
size(${name}$,${i+offset}$)
346+
#:endfor
347+
#:endcall
348+
#:enddef
349+
350+
#! Generates an array shape specifier from an N-D array of sizes
351+
#!
352+
#! Args:
353+
#! name (str): Name of the original variable
354+
#! rank (int): Rank of the original variable
355+
#! offset(int): optional offset of the dimension loop (default = 0)
356+
#!
357+
#! Returns:
358+
#! Array rank suffix string enclosed in braces
359+
#!
360+
#! E.g.,
361+
#! shape_from_array_data('mat', 5)}$
362+
#! -> (1:mat(1),1:mat(2),1:mat(3),1:mat(4),1:mat(5))
363+
#! shape_from_array_data('mat', 5, 2)}$
364+
#! -> (1:mat(3),1:mat(4),1:mat(5),1:mat(6),1:mat(7))
365+
#!
366+
#:def shape_from_array_data(name, rank, offset=0)
367+
#:assert rank > 0
368+
#:call join_lines(joinstr=", ", prefix="(", suffix=")")
369+
#:for i in range(1, rank + 1)
370+
1:${name}$(${i+offset}$)
371+
#:endfor
372+
#:endcall
373+
#:enddef
374+
375+
#!
376+
#! Start a sequence of loop with indexed variables over an N-D array
377+
#!
378+
#! Args:
379+
#! varname (str): Name of the variable to be used as prefix
380+
#! matname (str): Name of the variable to be used as array
381+
#! n (int): Number of nested loops to be created (1=innermost; n=outermost)
382+
#! dim_offset (int): Optional dimension offset (1st loop is over dimension 1+dim_offset)
383+
#! intent (str): Optional indentation. Default: 8 spaces
384+
#!
385+
#!
386+
#:def loop_variables_start(varname, matname, n, dim_offset=0, indent=" "*8)
387+
#:assert n > 0
388+
#:for i in range(1, n + 1)
389+
${indent}$do ${varname}$${n+1+dim_offset-i}$ = lbound(${matname}$, ${n+1+dim_offset-i}$), ubound(${matname}$, ${n+1+dim_offset-i}$)
390+
#:endfor
391+
#:enddef
392+
393+
#:def loop_variables_end(n, indent=" "*8)
394+
#:assert n > 0
395+
#:call join_lines(joinstr="; ",prefix=indent)
396+
#:for i in range(1, n + 1)
397+
enddo
398+
#:endfor
399+
#:endcall
400+
#:enddef
401+
301402
#:endmute

src/CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ set(fppFiles
3232
stdlib_linalg_determinant.fypp
3333
stdlib_linalg_qr.fypp
3434
stdlib_linalg_inverse.fypp
35+
stdlib_linalg_norms.fypp
3536
stdlib_linalg_state.fypp
3637
stdlib_linalg_svd.fypp
3738
stdlib_linalg_cholesky.fypp

0 commit comments

Comments
 (0)