1 /* String intrinsics helper functions.
2 Copyright 2002 Free Software Foundation, Inc.
3 Contributed by Paul Brook <paul@nowt.org>
5 This file is part of the GNU Fortran 95 runtime library (libgfor).
7 Libgfortran is free software; you can redistribute it and/or
8 modify it under the terms of the GNU Lesser General Public
9 License as published by the Free Software Foundation; either
10 version 2.1 of the License, or (at your option) any later version.
12 Libgfortran is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU Lesser General Public License for more details.
17 You should have received a copy of the GNU Lesser General Public
18 License along with libgfor; see the file COPYING.LIB. If not,
19 write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
20 Boston, MA 02111-1307, USA. */
23 /* Unlike what the name of this file suggests, we don't actually
24 implement the Fortran intrinsics here. At least, not with the
25 names they have in the standard. The functions here provide all
26 the support we need for the standard string intrinsics, and the
27 compiler translates the actual intrinsics calls to calls to
28 functions in this file. */
33 #include "libgfortran.h"
36 /* String functions. */
38 extern void copy_string (GFC_INTEGER_4
, char *, GFC_INTEGER_4
, const char *);
39 export_proto(copy_string
);
41 extern void concat_string (GFC_INTEGER_4
, char *,
42 GFC_INTEGER_4
, const char *,
43 GFC_INTEGER_4
, const char *);
44 export_proto(concat_string
);
46 extern GFC_INTEGER_4
string_len_trim (GFC_INTEGER_4
, const char *);
47 export_proto(string_len_trim
);
49 extern void adjustl (char *, GFC_INTEGER_4
, const char *);
50 export_proto(adjustl
);
52 extern void adjustr (char *, GFC_INTEGER_4
, const char *);
53 export_proto(adjustr
);
55 extern GFC_INTEGER_4
string_index (GFC_INTEGER_4
, const char *, GFC_INTEGER_4
,
56 const char *, GFC_LOGICAL_4
);
57 export_proto(string_index
);
59 extern GFC_INTEGER_4
string_scan (GFC_INTEGER_4
, const char *, GFC_INTEGER_4
,
60 const char *, GFC_LOGICAL_4
);
61 export_proto(string_scan
);
63 extern GFC_INTEGER_4
string_verify (GFC_INTEGER_4
, const char *, GFC_INTEGER_4
,
64 const char *, GFC_LOGICAL_4
);
65 export_proto(string_verify
);
67 extern void string_trim (GFC_INTEGER_4
*, void **, GFC_INTEGER_4
, const char *);
68 export_proto(string_trim
);
70 extern void string_repeat (char *, GFC_INTEGER_4
, const char *, GFC_INTEGER_4
);
71 export_proto(string_repeat
);
73 /* The two areas may overlap so we use memmove. */
76 copy_string (GFC_INTEGER_4 destlen
, char * dest
,
77 GFC_INTEGER_4 srclen
, const char * src
)
79 if (srclen
>= destlen
)
81 /* This will truncate if too long. */
82 memmove (dest
, src
, destlen
);
83 /*memcpy (dest, src, destlen);*/
87 memmove (dest
, src
, srclen
);
88 /*memcpy (dest, src, srclen);*/
89 /* Pad with spaces. */
90 memset (&dest
[srclen
], ' ', destlen
- srclen
);
95 /* Strings of unequal length are extended with pad characters. */
98 compare_string (GFC_INTEGER_4 len1
, const char * s1
,
99 GFC_INTEGER_4 len2
, const char * s2
)
105 res
= strncmp (s1
, s2
, (len1
< len2
) ? len1
: len2
);
139 iexport(compare_string
);
142 /* The destination and source should not overlap. */
145 concat_string (GFC_INTEGER_4 destlen
, char * dest
,
146 GFC_INTEGER_4 len1
, const char * s1
,
147 GFC_INTEGER_4 len2
, const char * s2
)
151 memcpy (dest
, s1
, destlen
);
154 memcpy (dest
, s1
, len1
);
160 memcpy (dest
, s2
, destlen
);
164 memcpy (dest
, s2
, len2
);
165 memset (&dest
[len2
], ' ', destlen
- len2
);
169 /* Return string with all trailing blanks removed. */
172 string_trim (GFC_INTEGER_4
* len
, void ** dest
, GFC_INTEGER_4 slen
,
177 /* Determine length of result string. */
178 for (i
= slen
- 1; i
>= 0; i
--)
187 /* Allocate space for result string. */
188 *dest
= internal_malloc_size (*len
);
190 /* copy string if necessary. */
191 memmove (*dest
, src
, *len
);
196 /* The length of a string not including trailing blanks. */
199 string_len_trim (GFC_INTEGER_4 len
, const char * s
)
203 for (i
= len
- 1; i
>= 0; i
--)
212 /* Find a substring within a string. */
215 string_index (GFC_INTEGER_4 slen
, const char * str
, GFC_INTEGER_4 sslen
,
216 const char * sstr
, GFC_LOGICAL_4 back
)
231 last
= slen
+ 1 - sslen
;
238 start
= slen
- sslen
;
242 for (; start
!= last
; start
+= delta
)
244 for (i
= 0; i
< sslen
; i
++)
246 if (str
[start
+ i
] != sstr
[i
])
256 /* Remove leading blanks from a string, padding at end. The src and dest
257 should not overlap. */
260 adjustl (char *dest
, GFC_INTEGER_4 len
, const char *src
)
265 while (i
<len
&& src
[i
] == ' ')
269 memcpy (dest
, &src
[i
], len
- i
);
271 memset (&dest
[len
- i
], ' ', i
);
275 /* Remove trailing blanks from a string. */
278 adjustr (char *dest
, GFC_INTEGER_4 len
, const char *src
)
283 while (i
> 0 && src
[i
- 1] == ' ')
287 memset (dest
, ' ', len
- i
);
288 memcpy (dest
+ (len
- i
), src
, i
);
292 /* Scan a string for any one of the characters in a set of characters. */
295 string_scan (GFC_INTEGER_4 slen
, const char * str
, GFC_INTEGER_4 setlen
,
296 const char * set
, GFC_LOGICAL_4 back
)
303 if (slen
== 0 || setlen
== 0)
320 for (; start
!= last
; start
+= delta
)
322 for (i
= 0; i
< setlen
; i
++)
324 if (str
[start
] == set
[i
])
333 /* Verify that a set of characters contains all the characters in a
334 string by indentifying the position of the first character in a
335 characters that dose not appear in a given set of characters. */
338 string_verify (GFC_INTEGER_4 slen
, const char * str
, GFC_INTEGER_4 setlen
,
339 const char * set
, GFC_LOGICAL_4 back
)
361 for (; start
!= last
; start
+= delta
)
363 for (i
= 0; i
< setlen
; i
++)
365 if (str
[start
] == set
[i
])
376 /* Concatenate several copies of a string. */
379 string_repeat (char * dest
, GFC_INTEGER_4 slen
,
380 const char * src
, GFC_INTEGER_4 ncopies
)
384 /* See if ncopies is valid. */
387 /* The error is already reported. */
388 runtime_error ("Augument NCOPIES is negative.");
391 /* Copy characters. */
392 for (i
= 0; i
< ncopies
; i
++)
394 memmove (dest
+ (i
* slen
), src
, slen
);