Linux Audio

Check our new training course

Loading...
v4.10.11
  1/* IIO - useful set of util functionality
  2 *
  3 * Copyright (c) 2008 Jonathan Cameron
  4 *
  5 * This program is free software; you can redistribute it and/or modify it
  6 * under the terms of the GNU General Public License version 2 as published by
  7 * the Free Software Foundation.
  8 */
  9#include <string.h>
 10#include <stdlib.h>
 11#include <stdio.h>
 12#include <stdint.h>
 13#include <dirent.h>
 14#include <errno.h>
 15#include <ctype.h>
 16#include "iio_utils.h"
 17
 18const char *iio_dir = "/sys/bus/iio/devices/";
 19
 20static char * const iio_direction[] = {
 21	"in",
 22	"out",
 23};
 24
 25/**
 26 * iioutils_break_up_name() - extract generic name from full channel name
 27 * @full_name: the full channel name
 28 * @generic_name: the output generic channel name
 29 *
 30 * Returns 0 on success, or a negative error code if string extraction failed.
 31 **/
 32int iioutils_break_up_name(const char *full_name, char **generic_name)
 33{
 34	char *current;
 35	char *w, *r;
 36	char *working, *prefix = "";
 37	int i, ret;
 38
 39	for (i = 0; i < ARRAY_SIZE(iio_direction); i++)
 40		if (!strncmp(full_name, iio_direction[i],
 41			     strlen(iio_direction[i]))) {
 42			prefix = iio_direction[i];
 43			break;
 44		}
 45
 46	current = strdup(full_name + strlen(prefix) + 1);
 47	if (!current)
 48		return -ENOMEM;
 49
 50	working = strtok(current, "_\0");
 51	if (!working) {
 52		free(current);
 53		return -EINVAL;
 54	}
 55
 56	w = working;
 57	r = working;
 58
 59	while (*r != '\0') {
 60		if (!isdigit(*r)) {
 61			*w = *r;
 62			w++;
 63		}
 64
 65		r++;
 66	}
 67	*w = '\0';
 68	ret = asprintf(generic_name, "%s_%s", prefix, working);
 69	free(current);
 70
 71	return (ret == -1) ? -ENOMEM : 0;
 72}
 73
 74/**
 75 * iioutils_get_type() - find and process _type attribute data
 76 * @is_signed: output whether channel is signed
 77 * @bytes: output how many bytes the channel storage occupies
 78 * @bits_used: output number of valid bits of data
 79 * @shift: output amount of bits to shift right data before applying bit mask
 80 * @mask: output a bit mask for the raw data
 81 * @be: output if data in big endian
 82 * @device_dir: the IIO device directory
 83 * @name: the channel name
 84 * @generic_name: the channel type name
 85 *
 86 * Returns a value >= 0 on success, otherwise a negative error code.
 87 **/
 88int iioutils_get_type(unsigned *is_signed, unsigned *bytes, unsigned *bits_used,
 89		      unsigned *shift, uint64_t *mask, unsigned *be,
 90		      const char *device_dir, const char *name,
 91		      const char *generic_name)
 92{
 93	FILE *sysfsfp;
 94	int ret;
 95	DIR *dp;
 96	char *scan_el_dir, *builtname, *builtname_generic, *filename = 0;
 97	char signchar, endianchar;
 98	unsigned padint;
 99	const struct dirent *ent;
100
101	ret = asprintf(&scan_el_dir, FORMAT_SCAN_ELEMENTS_DIR, device_dir);
102	if (ret < 0)
103		return -ENOMEM;
104
105	ret = asprintf(&builtname, FORMAT_TYPE_FILE, name);
106	if (ret < 0) {
107		ret = -ENOMEM;
108		goto error_free_scan_el_dir;
109	}
110	ret = asprintf(&builtname_generic, FORMAT_TYPE_FILE, generic_name);
111	if (ret < 0) {
112		ret = -ENOMEM;
113		goto error_free_builtname;
114	}
115
116	dp = opendir(scan_el_dir);
117	if (!dp) {
118		ret = -errno;
119		goto error_free_builtname_generic;
120	}
121
122	ret = -ENOENT;
123	while (ent = readdir(dp), ent)
 
 
 
 
124		if ((strcmp(builtname, ent->d_name) == 0) ||
125		    (strcmp(builtname_generic, ent->d_name) == 0)) {
126			ret = asprintf(&filename,
127				       "%s/%s", scan_el_dir, ent->d_name);
128			if (ret < 0) {
129				ret = -ENOMEM;
130				goto error_closedir;
131			}
132
133			sysfsfp = fopen(filename, "r");
134			if (!sysfsfp) {
135				ret = -errno;
136				fprintf(stderr, "failed to open %s\n",
137					filename);
138				goto error_free_filename;
139			}
140
141			ret = fscanf(sysfsfp,
142				     "%ce:%c%u/%u>>%u",
143				     &endianchar,
144				     &signchar,
145				     bits_used,
146				     &padint, shift);
147			if (ret < 0) {
148				ret = -errno;
149				fprintf(stderr,
150					"failed to pass scan type description\n");
151				goto error_close_sysfsfp;
152			} else if (ret != 5) {
153				ret = -EIO;
154				fprintf(stderr,
155					"scan type description didn't match\n");
156				goto error_close_sysfsfp;
157			}
158
159			*be = (endianchar == 'b');
160			*bytes = padint / 8;
161			if (*bits_used == 64)
162				*mask = ~0;
163			else
164				*mask = (1ULL << *bits_used) - 1;
165
166			*is_signed = (signchar == 's');
167			if (fclose(sysfsfp)) {
168				ret = -errno;
169				fprintf(stderr, "Failed to close %s\n",
170					filename);
171				goto error_free_filename;
172			}
173
174			sysfsfp = 0;
175			free(filename);
176			filename = 0;
177
178			/*
179			 * Avoid having a more generic entry overwriting
180			 * the settings.
181			 */
182			if (strcmp(builtname, ent->d_name) == 0)
183				break;
184		}
185
186error_close_sysfsfp:
187	if (sysfsfp)
188		if (fclose(sysfsfp))
189			perror("iioutils_get_type(): Failed to close file");
190
191error_free_filename:
192	if (filename)
193		free(filename);
194
195error_closedir:
196	if (closedir(dp) == -1)
197		perror("iioutils_get_type(): Failed to close directory");
198
199error_free_builtname_generic:
200	free(builtname_generic);
201error_free_builtname:
202	free(builtname);
203error_free_scan_el_dir:
204	free(scan_el_dir);
205
206	return ret;
207}
208
209/**
210 * iioutils_get_param_float() - read a float value from a channel parameter
211 * @output: output the float value
212 * @param_name: the parameter name to read
213 * @device_dir: the IIO device directory in sysfs
214 * @name: the channel name
215 * @generic_name: the channel type name
216 *
217 * Returns a value >= 0 on success, otherwise a negative error code.
218 **/
219int iioutils_get_param_float(float *output, const char *param_name,
220			     const char *device_dir, const char *name,
221			     const char *generic_name)
222{
223	FILE *sysfsfp;
224	int ret;
225	DIR *dp;
226	char *builtname, *builtname_generic;
227	char *filename = NULL;
228	const struct dirent *ent;
229
230	ret = asprintf(&builtname, "%s_%s", name, param_name);
231	if (ret < 0)
232		return -ENOMEM;
233
234	ret = asprintf(&builtname_generic,
235		       "%s_%s", generic_name, param_name);
236	if (ret < 0) {
237		ret = -ENOMEM;
238		goto error_free_builtname;
239	}
240
241	dp = opendir(device_dir);
242	if (!dp) {
243		ret = -errno;
244		goto error_free_builtname_generic;
245	}
246
247	ret = -ENOENT;
248	while (ent = readdir(dp), ent)
249		if ((strcmp(builtname, ent->d_name) == 0) ||
250		    (strcmp(builtname_generic, ent->d_name) == 0)) {
251			ret = asprintf(&filename,
252				       "%s/%s", device_dir, ent->d_name);
253			if (ret < 0) {
254				ret = -ENOMEM;
255				goto error_closedir;
256			}
257
258			sysfsfp = fopen(filename, "r");
259			if (!sysfsfp) {
260				ret = -errno;
261				goto error_free_filename;
262			}
263
264			errno = 0;
265			if (fscanf(sysfsfp, "%f", output) != 1)
266				ret = errno ? -errno : -ENODATA;
267
268			break;
269		}
270error_free_filename:
271	if (filename)
272		free(filename);
273
274error_closedir:
275	if (closedir(dp) == -1)
276		perror("iioutils_get_param_float(): Failed to close directory");
277
278error_free_builtname_generic:
279	free(builtname_generic);
280error_free_builtname:
281	free(builtname);
282
283	return ret;
284}
285
286/**
287 * bsort_channel_array_by_index() - sort the array in index order
288 * @ci_array: the iio_channel_info array to be sorted
289 * @cnt: the amount of array elements
290 **/
291
292void bsort_channel_array_by_index(struct iio_channel_info *ci_array, int cnt)
293{
294	struct iio_channel_info temp;
295	int x, y;
296
297	for (x = 0; x < cnt; x++)
298		for (y = 0; y < (cnt - 1); y++)
299			if (ci_array[y].index > ci_array[y + 1].index) {
300				temp = ci_array[y + 1];
301				ci_array[y + 1] = ci_array[y];
302				ci_array[y] = temp;
303			}
304}
305
306/**
307 * build_channel_array() - function to figure out what channels are present
308 * @device_dir: the IIO device directory in sysfs
309 * @ci_array: output the resulting array of iio_channel_info
310 * @counter: output the amount of array elements
311 *
312 * Returns 0 on success, otherwise a negative error code.
313 **/
314int build_channel_array(const char *device_dir,
315			struct iio_channel_info **ci_array, int *counter)
316{
317	DIR *dp;
318	FILE *sysfsfp;
319	int count = 0, i;
320	struct iio_channel_info *current;
321	int ret;
322	const struct dirent *ent;
323	char *scan_el_dir;
324	char *filename;
325
326	*counter = 0;
327	ret = asprintf(&scan_el_dir, FORMAT_SCAN_ELEMENTS_DIR, device_dir);
328	if (ret < 0)
329		return -ENOMEM;
330
331	dp = opendir(scan_el_dir);
332	if (!dp) {
333		ret = -errno;
334		goto error_free_name;
335	}
336
337	while (ent = readdir(dp), ent)
338		if (strcmp(ent->d_name + strlen(ent->d_name) - strlen("_en"),
339			   "_en") == 0) {
340			ret = asprintf(&filename,
341				       "%s/%s", scan_el_dir, ent->d_name);
342			if (ret < 0) {
343				ret = -ENOMEM;
344				goto error_close_dir;
345			}
346
347			sysfsfp = fopen(filename, "r");
348			if (!sysfsfp) {
349				ret = -errno;
350				free(filename);
351				goto error_close_dir;
352			}
353
354			errno = 0;
355			if (fscanf(sysfsfp, "%i", &ret) != 1) {
356				ret = errno ? -errno : -ENODATA;
357				if (fclose(sysfsfp))
358					perror("build_channel_array(): Failed to close file");
359
360				free(filename);
361				goto error_close_dir;
362			}
363			if (ret == 1)
364				(*counter)++;
365
366			if (fclose(sysfsfp)) {
367				ret = -errno;
368				free(filename);
369				goto error_close_dir;
370			}
371
372			free(filename);
373		}
374
375	*ci_array = malloc(sizeof(**ci_array) * (*counter));
376	if (!*ci_array) {
377		ret = -ENOMEM;
378		goto error_close_dir;
379	}
380
381	seekdir(dp, 0);
382	while (ent = readdir(dp), ent) {
383		if (strcmp(ent->d_name + strlen(ent->d_name) - strlen("_en"),
384			   "_en") == 0) {
385			int current_enabled = 0;
386
387			current = &(*ci_array)[count++];
388			ret = asprintf(&filename,
389				       "%s/%s", scan_el_dir, ent->d_name);
390			if (ret < 0) {
391				ret = -ENOMEM;
392				/* decrement count to avoid freeing name */
393				count--;
394				goto error_cleanup_array;
395			}
396
397			sysfsfp = fopen(filename, "r");
398			if (!sysfsfp) {
399				ret = -errno;
400				free(filename);
401				count--;
402				goto error_cleanup_array;
403			}
404
405			errno = 0;
406			if (fscanf(sysfsfp, "%i", &current_enabled) != 1) {
407				ret = errno ? -errno : -ENODATA;
408				free(filename);
409				count--;
410				goto error_cleanup_array;
411			}
412
413			if (fclose(sysfsfp)) {
414				ret = -errno;
415				free(filename);
416				count--;
417				goto error_cleanup_array;
418			}
419
420			if (!current_enabled) {
421				free(filename);
422				count--;
423				continue;
424			}
425
426			current->scale = 1.0;
427			current->offset = 0;
428			current->name = strndup(ent->d_name,
429						strlen(ent->d_name) -
430						strlen("_en"));
431			if (!current->name) {
432				free(filename);
433				ret = -ENOMEM;
434				count--;
435				goto error_cleanup_array;
436			}
437
438			/* Get the generic and specific name elements */
439			ret = iioutils_break_up_name(current->name,
440						     &current->generic_name);
441			if (ret) {
442				free(filename);
443				free(current->name);
444				count--;
445				goto error_cleanup_array;
446			}
447
448			ret = asprintf(&filename,
449				       "%s/%s_index",
450				       scan_el_dir,
451				       current->name);
452			if (ret < 0) {
453				free(filename);
454				ret = -ENOMEM;
455				goto error_cleanup_array;
456			}
457
458			sysfsfp = fopen(filename, "r");
459			if (!sysfsfp) {
460				ret = -errno;
461				fprintf(stderr, "failed to open %s\n",
462					filename);
463				free(filename);
464				goto error_cleanup_array;
465			}
466
467			errno = 0;
468			if (fscanf(sysfsfp, "%u", &current->index) != 1) {
469				ret = errno ? -errno : -ENODATA;
470				if (fclose(sysfsfp))
471					perror("build_channel_array(): Failed to close file");
472
473				free(filename);
474				goto error_cleanup_array;
475			}
476
477			if (fclose(sysfsfp)) {
478				ret = -errno;
479				free(filename);
480				goto error_cleanup_array;
481			}
482
483			free(filename);
484			/* Find the scale */
485			ret = iioutils_get_param_float(&current->scale,
486						       "scale",
487						       device_dir,
488						       current->name,
489						       current->generic_name);
490			if ((ret < 0) && (ret != -ENOENT))
491				goto error_cleanup_array;
492
493			ret = iioutils_get_param_float(&current->offset,
494						       "offset",
495						       device_dir,
496						       current->name,
497						       current->generic_name);
498			if ((ret < 0) && (ret != -ENOENT))
499				goto error_cleanup_array;
500
501			ret = iioutils_get_type(&current->is_signed,
502						&current->bytes,
503						&current->bits_used,
504						&current->shift,
505						&current->mask,
506						&current->be,
507						device_dir,
508						current->name,
509						current->generic_name);
510			if (ret < 0)
511				goto error_cleanup_array;
512		}
513	}
514
515	if (closedir(dp) == -1) {
516		ret = -errno;
517		goto error_cleanup_array;
518	}
519
520	free(scan_el_dir);
521	/* reorder so that the array is in index order */
522	bsort_channel_array_by_index(*ci_array, *counter);
523
524	return 0;
525
526error_cleanup_array:
527	for (i = count - 1;  i >= 0; i--) {
528		free((*ci_array)[i].name);
529		free((*ci_array)[i].generic_name);
530	}
531	free(*ci_array);
532	*ci_array = NULL;
533	*counter = 0;
534error_close_dir:
535	if (dp)
536		if (closedir(dp) == -1)
537			perror("build_channel_array(): Failed to close dir");
538
539error_free_name:
540	free(scan_el_dir);
541
542	return ret;
543}
544
545static int calc_digits(int num)
546{
547	int count = 0;
548
549	while (num != 0) {
550		num /= 10;
551		count++;
552	}
553
554	return count;
555}
556
557/**
558 * find_type_by_name() - function to match top level types by name
559 * @name: top level type instance name
560 * @type: the type of top level instance being searched
561 *
562 * Returns the device number of a matched IIO device on success, otherwise a
563 * negative error code.
564 * Typical types this is used for are device and trigger.
565 **/
566int find_type_by_name(const char *name, const char *type)
567{
568	const struct dirent *ent;
569	int number, numstrlen, ret;
570
571	FILE *namefp;
572	DIR *dp;
573	char thisname[IIO_MAX_NAME_LENGTH];
574	char *filename;
575
576	dp = opendir(iio_dir);
577	if (!dp) {
578		fprintf(stderr, "No industrialio devices available\n");
579		return -ENODEV;
580	}
581
582	while (ent = readdir(dp), ent) {
583		if (strcmp(ent->d_name, ".") != 0 &&
584		    strcmp(ent->d_name, "..") != 0 &&
585		    strlen(ent->d_name) > strlen(type) &&
586		    strncmp(ent->d_name, type, strlen(type)) == 0) {
587			errno = 0;
588			ret = sscanf(ent->d_name + strlen(type), "%d", &number);
589			if (ret < 0) {
590				ret = -errno;
591				fprintf(stderr,
592					"failed to read element number\n");
593				goto error_close_dir;
594			} else if (ret != 1) {
595				ret = -EIO;
596				fprintf(stderr,
597					"failed to match element number\n");
598				goto error_close_dir;
599			}
600
601			numstrlen = calc_digits(number);
602			/* verify the next character is not a colon */
603			if (strncmp(ent->d_name + strlen(type) + numstrlen,
604			    ":", 1) != 0) {
605				filename = malloc(strlen(iio_dir) + strlen(type)
606						  + numstrlen + 6);
607				if (!filename) {
608					ret = -ENOMEM;
609					goto error_close_dir;
610				}
611
612				ret = sprintf(filename, "%s%s%d/name", iio_dir,
613					      type, number);
614				if (ret < 0) {
615					free(filename);
616					goto error_close_dir;
617				}
618
619				namefp = fopen(filename, "r");
620				if (!namefp) {
621					free(filename);
622					continue;
623				}
624
625				free(filename);
626				errno = 0;
627				if (fscanf(namefp, "%s", thisname) != 1) {
628					ret = errno ? -errno : -ENODATA;
629					goto error_close_dir;
630				}
631
632				if (fclose(namefp)) {
633					ret = -errno;
634					goto error_close_dir;
635				}
636
637				if (strcmp(name, thisname) == 0) {
638					if (closedir(dp) == -1)
639						return -errno;
640
641					return number;
642				}
643			}
644		}
645	}
646	if (closedir(dp) == -1)
647		return -errno;
648
649	return -ENODEV;
650
651error_close_dir:
652	if (closedir(dp) == -1)
653		perror("find_type_by_name(): Failed to close directory");
654
655	return ret;
656}
657
658static int _write_sysfs_int(const char *filename, const char *basedir, int val,
659			    int verify)
660{
661	int ret = 0;
662	FILE *sysfsfp;
663	int test;
664	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
665
666	if (!temp)
667		return -ENOMEM;
668
669	ret = sprintf(temp, "%s/%s", basedir, filename);
670	if (ret < 0)
671		goto error_free;
672
673	sysfsfp = fopen(temp, "w");
674	if (!sysfsfp) {
675		ret = -errno;
676		fprintf(stderr, "failed to open %s\n", temp);
677		goto error_free;
678	}
679
680	ret = fprintf(sysfsfp, "%d", val);
681	if (ret < 0) {
682		if (fclose(sysfsfp))
683			perror("_write_sysfs_int(): Failed to close dir");
684
685		goto error_free;
686	}
687
688	if (fclose(sysfsfp)) {
689		ret = -errno;
690		goto error_free;
691	}
692
693	if (verify) {
694		sysfsfp = fopen(temp, "r");
695		if (!sysfsfp) {
696			ret = -errno;
697			fprintf(stderr, "failed to open %s\n", temp);
698			goto error_free;
699		}
700
701		if (fscanf(sysfsfp, "%d", &test) != 1) {
702			ret = errno ? -errno : -ENODATA;
703			if (fclose(sysfsfp))
704				perror("_write_sysfs_int(): Failed to close dir");
705
706			goto error_free;
707		}
708
709		if (fclose(sysfsfp)) {
710			ret = -errno;
711			goto error_free;
712		}
713
714		if (test != val) {
715			fprintf(stderr,
716				"Possible failure in int write %d to %s/%s\n",
717				val, basedir, filename);
718			ret = -1;
719		}
720	}
721
722error_free:
723	free(temp);
724	return ret;
725}
726
727/**
728 * write_sysfs_int() - write an integer value to a sysfs file
729 * @filename: name of the file to write to
730 * @basedir: the sysfs directory in which the file is to be found
731 * @val: integer value to write to file
732 *
733 * Returns a value >= 0 on success, otherwise a negative error code.
734 **/
735int write_sysfs_int(const char *filename, const char *basedir, int val)
736{
737	return _write_sysfs_int(filename, basedir, val, 0);
738}
739
740/**
741 * write_sysfs_int_and_verify() - write an integer value to a sysfs file
742 *				  and verify
743 * @filename: name of the file to write to
744 * @basedir: the sysfs directory in which the file is to be found
745 * @val: integer value to write to file
746 *
747 * Returns a value >= 0 on success, otherwise a negative error code.
748 **/
749int write_sysfs_int_and_verify(const char *filename, const char *basedir,
750			       int val)
751{
752	return _write_sysfs_int(filename, basedir, val, 1);
753}
754
755static int _write_sysfs_string(const char *filename, const char *basedir,
756			       const char *val, int verify)
757{
758	int ret = 0;
759	FILE  *sysfsfp;
760	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
761
762	if (!temp) {
763		fprintf(stderr, "Memory allocation failed\n");
764		return -ENOMEM;
765	}
766
767	ret = sprintf(temp, "%s/%s", basedir, filename);
768	if (ret < 0)
769		goto error_free;
770
771	sysfsfp = fopen(temp, "w");
772	if (!sysfsfp) {
773		ret = -errno;
774		fprintf(stderr, "Could not open %s\n", temp);
775		goto error_free;
776	}
777
778	ret = fprintf(sysfsfp, "%s", val);
779	if (ret < 0) {
780		if (fclose(sysfsfp))
781			perror("_write_sysfs_string(): Failed to close dir");
782
783		goto error_free;
784	}
785
786	if (fclose(sysfsfp)) {
787		ret = -errno;
788		goto error_free;
789	}
790
791	if (verify) {
792		sysfsfp = fopen(temp, "r");
793		if (!sysfsfp) {
794			ret = -errno;
795			fprintf(stderr, "Could not open file to verify\n");
796			goto error_free;
797		}
798
799		if (fscanf(sysfsfp, "%s", temp) != 1) {
800			ret = errno ? -errno : -ENODATA;
801			if (fclose(sysfsfp))
802				perror("_write_sysfs_string(): Failed to close dir");
803
804			goto error_free;
805		}
806
807		if (fclose(sysfsfp)) {
808			ret = -errno;
809			goto error_free;
810		}
811
812		if (strcmp(temp, val) != 0) {
813			fprintf(stderr,
814				"Possible failure in string write of %s "
815				"Should be %s written to %s/%s\n", temp, val,
816				basedir, filename);
817			ret = -1;
818		}
819	}
820
821error_free:
822	free(temp);
823
824	return ret;
825}
826
827/**
828 * write_sysfs_string_and_verify() - string write, readback and verify
829 * @filename: name of file to write to
830 * @basedir: the sysfs directory in which the file is to be found
831 * @val: the string to write
832 *
833 * Returns a value >= 0 on success, otherwise a negative error code.
834 **/
835int write_sysfs_string_and_verify(const char *filename, const char *basedir,
836				  const char *val)
837{
838	return _write_sysfs_string(filename, basedir, val, 1);
839}
840
841/**
842 * write_sysfs_string() - write string to a sysfs file
843 * @filename: name of file to write to
844 * @basedir: the sysfs directory in which the file is to be found
845 * @val: the string to write
846 *
847 * Returns a value >= 0 on success, otherwise a negative error code.
848 **/
849int write_sysfs_string(const char *filename, const char *basedir,
850		       const char *val)
851{
852	return _write_sysfs_string(filename, basedir, val, 0);
853}
854
855/**
856 * read_sysfs_posint() - read an integer value from file
857 * @filename: name of file to read from
858 * @basedir: the sysfs directory in which the file is to be found
859 *
860 * Returns the read integer value >= 0 on success, otherwise a negative error
861 * code.
862 **/
863int read_sysfs_posint(const char *filename, const char *basedir)
864{
865	int ret;
866	FILE  *sysfsfp;
867	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
868
869	if (!temp) {
870		fprintf(stderr, "Memory allocation failed");
871		return -ENOMEM;
872	}
873
874	ret = sprintf(temp, "%s/%s", basedir, filename);
875	if (ret < 0)
876		goto error_free;
877
878	sysfsfp = fopen(temp, "r");
879	if (!sysfsfp) {
880		ret = -errno;
881		goto error_free;
882	}
883
884	errno = 0;
885	if (fscanf(sysfsfp, "%d\n", &ret) != 1) {
886		ret = errno ? -errno : -ENODATA;
887		if (fclose(sysfsfp))
888			perror("read_sysfs_posint(): Failed to close dir");
889
890		goto error_free;
891	}
892
893	if (fclose(sysfsfp))
894		ret = -errno;
895
896error_free:
897	free(temp);
898
899	return ret;
900}
901
902/**
903 * read_sysfs_float() - read a float value from file
904 * @filename: name of file to read from
905 * @basedir: the sysfs directory in which the file is to be found
906 * @val: output the read float value
907 *
908 * Returns a value >= 0 on success, otherwise a negative error code.
909 **/
910int read_sysfs_float(const char *filename, const char *basedir, float *val)
911{
912	int ret = 0;
913	FILE  *sysfsfp;
914	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
915
916	if (!temp) {
917		fprintf(stderr, "Memory allocation failed");
918		return -ENOMEM;
919	}
920
921	ret = sprintf(temp, "%s/%s", basedir, filename);
922	if (ret < 0)
923		goto error_free;
924
925	sysfsfp = fopen(temp, "r");
926	if (!sysfsfp) {
927		ret = -errno;
928		goto error_free;
929	}
930
931	errno = 0;
932	if (fscanf(sysfsfp, "%f\n", val) != 1) {
933		ret = errno ? -errno : -ENODATA;
934		if (fclose(sysfsfp))
935			perror("read_sysfs_float(): Failed to close dir");
936
937		goto error_free;
938	}
939
940	if (fclose(sysfsfp))
941		ret = -errno;
942
943error_free:
944	free(temp);
945
946	return ret;
947}
948
949/**
950 * read_sysfs_string() - read a string from file
951 * @filename: name of file to read from
952 * @basedir: the sysfs directory in which the file is to be found
953 * @str: output the read string
954 *
955 * Returns a value >= 0 on success, otherwise a negative error code.
956 **/
957int read_sysfs_string(const char *filename, const char *basedir, char *str)
958{
959	int ret = 0;
960	FILE  *sysfsfp;
961	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
962
963	if (!temp) {
964		fprintf(stderr, "Memory allocation failed");
965		return -ENOMEM;
966	}
967
968	ret = sprintf(temp, "%s/%s", basedir, filename);
969	if (ret < 0)
970		goto error_free;
971
972	sysfsfp = fopen(temp, "r");
973	if (!sysfsfp) {
974		ret = -errno;
975		goto error_free;
976	}
977
978	errno = 0;
979	if (fscanf(sysfsfp, "%s\n", str) != 1) {
980		ret = errno ? -errno : -ENODATA;
981		if (fclose(sysfsfp))
982			perror("read_sysfs_string(): Failed to close dir");
983
984		goto error_free;
985	}
986
987	if (fclose(sysfsfp))
988		ret = -errno;
989
990error_free:
991	free(temp);
992
993	return ret;
994}
v4.6
  1/* IIO - useful set of util functionality
  2 *
  3 * Copyright (c) 2008 Jonathan Cameron
  4 *
  5 * This program is free software; you can redistribute it and/or modify it
  6 * under the terms of the GNU General Public License version 2 as published by
  7 * the Free Software Foundation.
  8 */
  9#include <string.h>
 10#include <stdlib.h>
 11#include <stdio.h>
 12#include <stdint.h>
 13#include <dirent.h>
 14#include <errno.h>
 15#include <ctype.h>
 16#include "iio_utils.h"
 17
 18const char *iio_dir = "/sys/bus/iio/devices/";
 19
 20static char * const iio_direction[] = {
 21	"in",
 22	"out",
 23};
 24
 25/**
 26 * iioutils_break_up_name() - extract generic name from full channel name
 27 * @full_name: the full channel name
 28 * @generic_name: the output generic channel name
 29 *
 30 * Returns 0 on success, or a negative error code if string extraction failed.
 31 **/
 32int iioutils_break_up_name(const char *full_name, char **generic_name)
 33{
 34	char *current;
 35	char *w, *r;
 36	char *working, *prefix = "";
 37	int i, ret;
 38
 39	for (i = 0; i < ARRAY_SIZE(iio_direction); i++)
 40		if (!strncmp(full_name, iio_direction[i],
 41			     strlen(iio_direction[i]))) {
 42			prefix = iio_direction[i];
 43			break;
 44		}
 45
 46	current = strdup(full_name + strlen(prefix) + 1);
 47	if (!current)
 48		return -ENOMEM;
 49
 50	working = strtok(current, "_\0");
 51	if (!working) {
 52		free(current);
 53		return -EINVAL;
 54	}
 55
 56	w = working;
 57	r = working;
 58
 59	while (*r != '\0') {
 60		if (!isdigit(*r)) {
 61			*w = *r;
 62			w++;
 63		}
 64
 65		r++;
 66	}
 67	*w = '\0';
 68	ret = asprintf(generic_name, "%s_%s", prefix, working);
 69	free(current);
 70
 71	return (ret == -1) ? -ENOMEM : 0;
 72}
 73
 74/**
 75 * iioutils_get_type() - find and process _type attribute data
 76 * @is_signed: output whether channel is signed
 77 * @bytes: output how many bytes the channel storage occupies
 78 * @bits_used: output number of valid bits of data
 79 * @shift: output amount of bits to shift right data before applying bit mask
 80 * @mask: output a bit mask for the raw data
 81 * @be: output if data in big endian
 82 * @device_dir: the IIO device directory
 83 * @name: the channel name
 84 * @generic_name: the channel type name
 85 *
 86 * Returns a value >= 0 on success, otherwise a negative error code.
 87 **/
 88int iioutils_get_type(unsigned *is_signed, unsigned *bytes, unsigned *bits_used,
 89		      unsigned *shift, uint64_t *mask, unsigned *be,
 90		      const char *device_dir, const char *name,
 91		      const char *generic_name)
 92{
 93	FILE *sysfsfp;
 94	int ret;
 95	DIR *dp;
 96	char *scan_el_dir, *builtname, *builtname_generic, *filename = 0;
 97	char signchar, endianchar;
 98	unsigned padint;
 99	const struct dirent *ent;
100
101	ret = asprintf(&scan_el_dir, FORMAT_SCAN_ELEMENTS_DIR, device_dir);
102	if (ret < 0)
103		return -ENOMEM;
104
105	ret = asprintf(&builtname, FORMAT_TYPE_FILE, name);
106	if (ret < 0) {
107		ret = -ENOMEM;
108		goto error_free_scan_el_dir;
109	}
110	ret = asprintf(&builtname_generic, FORMAT_TYPE_FILE, generic_name);
111	if (ret < 0) {
112		ret = -ENOMEM;
113		goto error_free_builtname;
114	}
115
116	dp = opendir(scan_el_dir);
117	if (!dp) {
118		ret = -errno;
119		goto error_free_builtname_generic;
120	}
121
122	ret = -ENOENT;
123	while (ent = readdir(dp), ent)
124		/*
125		 * Do we allow devices to override a generic name with
126		 * a specific one?
127		 */
128		if ((strcmp(builtname, ent->d_name) == 0) ||
129		    (strcmp(builtname_generic, ent->d_name) == 0)) {
130			ret = asprintf(&filename,
131				       "%s/%s", scan_el_dir, ent->d_name);
132			if (ret < 0) {
133				ret = -ENOMEM;
134				goto error_closedir;
135			}
136
137			sysfsfp = fopen(filename, "r");
138			if (!sysfsfp) {
139				ret = -errno;
140				fprintf(stderr, "failed to open %s\n",
141					filename);
142				goto error_free_filename;
143			}
144
145			ret = fscanf(sysfsfp,
146				     "%ce:%c%u/%u>>%u",
147				     &endianchar,
148				     &signchar,
149				     bits_used,
150				     &padint, shift);
151			if (ret < 0) {
152				ret = -errno;
153				fprintf(stderr,
154					"failed to pass scan type description\n");
155				goto error_close_sysfsfp;
156			} else if (ret != 5) {
157				ret = -EIO;
158				fprintf(stderr,
159					"scan type description didn't match\n");
160				goto error_close_sysfsfp;
161			}
162
163			*be = (endianchar == 'b');
164			*bytes = padint / 8;
165			if (*bits_used == 64)
166				*mask = ~0;
167			else
168				*mask = (1ULL << *bits_used) - 1;
169
170			*is_signed = (signchar == 's');
171			if (fclose(sysfsfp)) {
172				ret = -errno;
173				fprintf(stderr, "Failed to close %s\n",
174					filename);
175				goto error_free_filename;
176			}
177
178			sysfsfp = 0;
179			free(filename);
180			filename = 0;
 
 
 
 
 
 
 
181		}
182
183error_close_sysfsfp:
184	if (sysfsfp)
185		if (fclose(sysfsfp))
186			perror("iioutils_get_type(): Failed to close file");
187
188error_free_filename:
189	if (filename)
190		free(filename);
191
192error_closedir:
193	if (closedir(dp) == -1)
194		perror("iioutils_get_type(): Failed to close directory");
195
196error_free_builtname_generic:
197	free(builtname_generic);
198error_free_builtname:
199	free(builtname);
200error_free_scan_el_dir:
201	free(scan_el_dir);
202
203	return ret;
204}
205
206/**
207 * iioutils_get_param_float() - read a float value from a channel parameter
208 * @output: output the float value
209 * @param_name: the parameter name to read
210 * @device_dir: the IIO device directory in sysfs
211 * @name: the channel name
212 * @generic_name: the channel type name
213 *
214 * Returns a value >= 0 on success, otherwise a negative error code.
215 **/
216int iioutils_get_param_float(float *output, const char *param_name,
217			     const char *device_dir, const char *name,
218			     const char *generic_name)
219{
220	FILE *sysfsfp;
221	int ret;
222	DIR *dp;
223	char *builtname, *builtname_generic;
224	char *filename = NULL;
225	const struct dirent *ent;
226
227	ret = asprintf(&builtname, "%s_%s", name, param_name);
228	if (ret < 0)
229		return -ENOMEM;
230
231	ret = asprintf(&builtname_generic,
232		       "%s_%s", generic_name, param_name);
233	if (ret < 0) {
234		ret = -ENOMEM;
235		goto error_free_builtname;
236	}
237
238	dp = opendir(device_dir);
239	if (!dp) {
240		ret = -errno;
241		goto error_free_builtname_generic;
242	}
243
244	ret = -ENOENT;
245	while (ent = readdir(dp), ent)
246		if ((strcmp(builtname, ent->d_name) == 0) ||
247		    (strcmp(builtname_generic, ent->d_name) == 0)) {
248			ret = asprintf(&filename,
249				       "%s/%s", device_dir, ent->d_name);
250			if (ret < 0) {
251				ret = -ENOMEM;
252				goto error_closedir;
253			}
254
255			sysfsfp = fopen(filename, "r");
256			if (!sysfsfp) {
257				ret = -errno;
258				goto error_free_filename;
259			}
260
261			errno = 0;
262			if (fscanf(sysfsfp, "%f", output) != 1)
263				ret = errno ? -errno : -ENODATA;
264
265			break;
266		}
267error_free_filename:
268	if (filename)
269		free(filename);
270
271error_closedir:
272	if (closedir(dp) == -1)
273		perror("iioutils_get_param_float(): Failed to close directory");
274
275error_free_builtname_generic:
276	free(builtname_generic);
277error_free_builtname:
278	free(builtname);
279
280	return ret;
281}
282
283/**
284 * bsort_channel_array_by_index() - sort the array in index order
285 * @ci_array: the iio_channel_info array to be sorted
286 * @cnt: the amount of array elements
287 **/
288
289void bsort_channel_array_by_index(struct iio_channel_info *ci_array, int cnt)
290{
291	struct iio_channel_info temp;
292	int x, y;
293
294	for (x = 0; x < cnt; x++)
295		for (y = 0; y < (cnt - 1); y++)
296			if (ci_array[y].index > ci_array[y + 1].index) {
297				temp = ci_array[y + 1];
298				ci_array[y + 1] = ci_array[y];
299				ci_array[y] = temp;
300			}
301}
302
303/**
304 * build_channel_array() - function to figure out what channels are present
305 * @device_dir: the IIO device directory in sysfs
306 * @ci_array: output the resulting array of iio_channel_info
307 * @counter: output the amount of array elements
308 *
309 * Returns 0 on success, otherwise a negative error code.
310 **/
311int build_channel_array(const char *device_dir,
312			struct iio_channel_info **ci_array, int *counter)
313{
314	DIR *dp;
315	FILE *sysfsfp;
316	int count = 0, i;
317	struct iio_channel_info *current;
318	int ret;
319	const struct dirent *ent;
320	char *scan_el_dir;
321	char *filename;
322
323	*counter = 0;
324	ret = asprintf(&scan_el_dir, FORMAT_SCAN_ELEMENTS_DIR, device_dir);
325	if (ret < 0)
326		return -ENOMEM;
327
328	dp = opendir(scan_el_dir);
329	if (!dp) {
330		ret = -errno;
331		goto error_free_name;
332	}
333
334	while (ent = readdir(dp), ent)
335		if (strcmp(ent->d_name + strlen(ent->d_name) - strlen("_en"),
336			   "_en") == 0) {
337			ret = asprintf(&filename,
338				       "%s/%s", scan_el_dir, ent->d_name);
339			if (ret < 0) {
340				ret = -ENOMEM;
341				goto error_close_dir;
342			}
343
344			sysfsfp = fopen(filename, "r");
345			if (!sysfsfp) {
346				ret = -errno;
347				free(filename);
348				goto error_close_dir;
349			}
350
351			errno = 0;
352			if (fscanf(sysfsfp, "%i", &ret) != 1) {
353				ret = errno ? -errno : -ENODATA;
354				if (fclose(sysfsfp))
355					perror("build_channel_array(): Failed to close file");
356
357				free(filename);
358				goto error_close_dir;
359			}
360			if (ret == 1)
361				(*counter)++;
362
363			if (fclose(sysfsfp)) {
364				ret = -errno;
365				free(filename);
366				goto error_close_dir;
367			}
368
369			free(filename);
370		}
371
372	*ci_array = malloc(sizeof(**ci_array) * (*counter));
373	if (!*ci_array) {
374		ret = -ENOMEM;
375		goto error_close_dir;
376	}
377
378	seekdir(dp, 0);
379	while (ent = readdir(dp), ent) {
380		if (strcmp(ent->d_name + strlen(ent->d_name) - strlen("_en"),
381			   "_en") == 0) {
382			int current_enabled = 0;
383
384			current = &(*ci_array)[count++];
385			ret = asprintf(&filename,
386				       "%s/%s", scan_el_dir, ent->d_name);
387			if (ret < 0) {
388				ret = -ENOMEM;
389				/* decrement count to avoid freeing name */
390				count--;
391				goto error_cleanup_array;
392			}
393
394			sysfsfp = fopen(filename, "r");
395			if (!sysfsfp) {
396				ret = -errno;
397				free(filename);
398				count--;
399				goto error_cleanup_array;
400			}
401
402			errno = 0;
403			if (fscanf(sysfsfp, "%i", &current_enabled) != 1) {
404				ret = errno ? -errno : -ENODATA;
405				free(filename);
406				count--;
407				goto error_cleanup_array;
408			}
409
410			if (fclose(sysfsfp)) {
411				ret = -errno;
412				free(filename);
413				count--;
414				goto error_cleanup_array;
415			}
416
417			if (!current_enabled) {
418				free(filename);
419				count--;
420				continue;
421			}
422
423			current->scale = 1.0;
424			current->offset = 0;
425			current->name = strndup(ent->d_name,
426						strlen(ent->d_name) -
427						strlen("_en"));
428			if (!current->name) {
429				free(filename);
430				ret = -ENOMEM;
431				count--;
432				goto error_cleanup_array;
433			}
434
435			/* Get the generic and specific name elements */
436			ret = iioutils_break_up_name(current->name,
437						     &current->generic_name);
438			if (ret) {
439				free(filename);
440				free(current->name);
441				count--;
442				goto error_cleanup_array;
443			}
444
445			ret = asprintf(&filename,
446				       "%s/%s_index",
447				       scan_el_dir,
448				       current->name);
449			if (ret < 0) {
450				free(filename);
451				ret = -ENOMEM;
452				goto error_cleanup_array;
453			}
454
455			sysfsfp = fopen(filename, "r");
456			if (!sysfsfp) {
457				ret = -errno;
458				fprintf(stderr, "failed to open %s\n",
459					filename);
460				free(filename);
461				goto error_cleanup_array;
462			}
463
464			errno = 0;
465			if (fscanf(sysfsfp, "%u", &current->index) != 1) {
466				ret = errno ? -errno : -ENODATA;
467				if (fclose(sysfsfp))
468					perror("build_channel_array(): Failed to close file");
469
470				free(filename);
471				goto error_cleanup_array;
472			}
473
474			if (fclose(sysfsfp)) {
475				ret = -errno;
476				free(filename);
477				goto error_cleanup_array;
478			}
479
480			free(filename);
481			/* Find the scale */
482			ret = iioutils_get_param_float(&current->scale,
483						       "scale",
484						       device_dir,
485						       current->name,
486						       current->generic_name);
487			if ((ret < 0) && (ret != -ENOENT))
488				goto error_cleanup_array;
489
490			ret = iioutils_get_param_float(&current->offset,
491						       "offset",
492						       device_dir,
493						       current->name,
494						       current->generic_name);
495			if ((ret < 0) && (ret != -ENOENT))
496				goto error_cleanup_array;
497
498			ret = iioutils_get_type(&current->is_signed,
499						&current->bytes,
500						&current->bits_used,
501						&current->shift,
502						&current->mask,
503						&current->be,
504						device_dir,
505						current->name,
506						current->generic_name);
507			if (ret < 0)
508				goto error_cleanup_array;
509		}
510	}
511
512	if (closedir(dp) == -1) {
513		ret = -errno;
514		goto error_cleanup_array;
515	}
516
517	free(scan_el_dir);
518	/* reorder so that the array is in index order */
519	bsort_channel_array_by_index(*ci_array, *counter);
520
521	return 0;
522
523error_cleanup_array:
524	for (i = count - 1;  i >= 0; i--) {
525		free((*ci_array)[i].name);
526		free((*ci_array)[i].generic_name);
527	}
528	free(*ci_array);
529	*ci_array = NULL;
530	*counter = 0;
531error_close_dir:
532	if (dp)
533		if (closedir(dp) == -1)
534			perror("build_channel_array(): Failed to close dir");
535
536error_free_name:
537	free(scan_el_dir);
538
539	return ret;
540}
541
542static int calc_digits(int num)
543{
544	int count = 0;
545
546	while (num != 0) {
547		num /= 10;
548		count++;
549	}
550
551	return count;
552}
553
554/**
555 * find_type_by_name() - function to match top level types by name
556 * @name: top level type instance name
557 * @type: the type of top level instance being searched
558 *
559 * Returns the device number of a matched IIO device on success, otherwise a
560 * negative error code.
561 * Typical types this is used for are device and trigger.
562 **/
563int find_type_by_name(const char *name, const char *type)
564{
565	const struct dirent *ent;
566	int number, numstrlen, ret;
567
568	FILE *namefp;
569	DIR *dp;
570	char thisname[IIO_MAX_NAME_LENGTH];
571	char *filename;
572
573	dp = opendir(iio_dir);
574	if (!dp) {
575		fprintf(stderr, "No industrialio devices available\n");
576		return -ENODEV;
577	}
578
579	while (ent = readdir(dp), ent) {
580		if (strcmp(ent->d_name, ".") != 0 &&
581		    strcmp(ent->d_name, "..") != 0 &&
582		    strlen(ent->d_name) > strlen(type) &&
583		    strncmp(ent->d_name, type, strlen(type)) == 0) {
584			errno = 0;
585			ret = sscanf(ent->d_name + strlen(type), "%d", &number);
586			if (ret < 0) {
587				ret = -errno;
588				fprintf(stderr,
589					"failed to read element number\n");
590				goto error_close_dir;
591			} else if (ret != 1) {
592				ret = -EIO;
593				fprintf(stderr,
594					"failed to match element number\n");
595				goto error_close_dir;
596			}
597
598			numstrlen = calc_digits(number);
599			/* verify the next character is not a colon */
600			if (strncmp(ent->d_name + strlen(type) + numstrlen,
601			    ":", 1) != 0) {
602				filename = malloc(strlen(iio_dir) + strlen(type)
603						  + numstrlen + 6);
604				if (!filename) {
605					ret = -ENOMEM;
606					goto error_close_dir;
607				}
608
609				ret = sprintf(filename, "%s%s%d/name", iio_dir,
610					      type, number);
611				if (ret < 0) {
612					free(filename);
613					goto error_close_dir;
614				}
615
616				namefp = fopen(filename, "r");
617				if (!namefp) {
618					free(filename);
619					continue;
620				}
621
622				free(filename);
623				errno = 0;
624				if (fscanf(namefp, "%s", thisname) != 1) {
625					ret = errno ? -errno : -ENODATA;
626					goto error_close_dir;
627				}
628
629				if (fclose(namefp)) {
630					ret = -errno;
631					goto error_close_dir;
632				}
633
634				if (strcmp(name, thisname) == 0) {
635					if (closedir(dp) == -1)
636						return -errno;
637
638					return number;
639				}
640			}
641		}
642	}
643	if (closedir(dp) == -1)
644		return -errno;
645
646	return -ENODEV;
647
648error_close_dir:
649	if (closedir(dp) == -1)
650		perror("find_type_by_name(): Failed to close directory");
651
652	return ret;
653}
654
655static int _write_sysfs_int(const char *filename, const char *basedir, int val,
656			    int verify)
657{
658	int ret = 0;
659	FILE *sysfsfp;
660	int test;
661	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
662
663	if (!temp)
664		return -ENOMEM;
665
666	ret = sprintf(temp, "%s/%s", basedir, filename);
667	if (ret < 0)
668		goto error_free;
669
670	sysfsfp = fopen(temp, "w");
671	if (!sysfsfp) {
672		ret = -errno;
673		fprintf(stderr, "failed to open %s\n", temp);
674		goto error_free;
675	}
676
677	ret = fprintf(sysfsfp, "%d", val);
678	if (ret < 0) {
679		if (fclose(sysfsfp))
680			perror("_write_sysfs_int(): Failed to close dir");
681
682		goto error_free;
683	}
684
685	if (fclose(sysfsfp)) {
686		ret = -errno;
687		goto error_free;
688	}
689
690	if (verify) {
691		sysfsfp = fopen(temp, "r");
692		if (!sysfsfp) {
693			ret = -errno;
694			fprintf(stderr, "failed to open %s\n", temp);
695			goto error_free;
696		}
697
698		if (fscanf(sysfsfp, "%d", &test) != 1) {
699			ret = errno ? -errno : -ENODATA;
700			if (fclose(sysfsfp))
701				perror("_write_sysfs_int(): Failed to close dir");
702
703			goto error_free;
704		}
705
706		if (fclose(sysfsfp)) {
707			ret = -errno;
708			goto error_free;
709		}
710
711		if (test != val) {
712			fprintf(stderr,
713				"Possible failure in int write %d to %s/%s\n",
714				val, basedir, filename);
715			ret = -1;
716		}
717	}
718
719error_free:
720	free(temp);
721	return ret;
722}
723
724/**
725 * write_sysfs_int() - write an integer value to a sysfs file
726 * @filename: name of the file to write to
727 * @basedir: the sysfs directory in which the file is to be found
728 * @val: integer value to write to file
729 *
730 * Returns a value >= 0 on success, otherwise a negative error code.
731 **/
732int write_sysfs_int(const char *filename, const char *basedir, int val)
733{
734	return _write_sysfs_int(filename, basedir, val, 0);
735}
736
737/**
738 * write_sysfs_int_and_verify() - write an integer value to a sysfs file
739 *				  and verify
740 * @filename: name of the file to write to
741 * @basedir: the sysfs directory in which the file is to be found
742 * @val: integer value to write to file
743 *
744 * Returns a value >= 0 on success, otherwise a negative error code.
745 **/
746int write_sysfs_int_and_verify(const char *filename, const char *basedir,
747			       int val)
748{
749	return _write_sysfs_int(filename, basedir, val, 1);
750}
751
752static int _write_sysfs_string(const char *filename, const char *basedir,
753			       const char *val, int verify)
754{
755	int ret = 0;
756	FILE  *sysfsfp;
757	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
758
759	if (!temp) {
760		fprintf(stderr, "Memory allocation failed\n");
761		return -ENOMEM;
762	}
763
764	ret = sprintf(temp, "%s/%s", basedir, filename);
765	if (ret < 0)
766		goto error_free;
767
768	sysfsfp = fopen(temp, "w");
769	if (!sysfsfp) {
770		ret = -errno;
771		fprintf(stderr, "Could not open %s\n", temp);
772		goto error_free;
773	}
774
775	ret = fprintf(sysfsfp, "%s", val);
776	if (ret < 0) {
777		if (fclose(sysfsfp))
778			perror("_write_sysfs_string(): Failed to close dir");
779
780		goto error_free;
781	}
782
783	if (fclose(sysfsfp)) {
784		ret = -errno;
785		goto error_free;
786	}
787
788	if (verify) {
789		sysfsfp = fopen(temp, "r");
790		if (!sysfsfp) {
791			ret = -errno;
792			fprintf(stderr, "Could not open file to verify\n");
793			goto error_free;
794		}
795
796		if (fscanf(sysfsfp, "%s", temp) != 1) {
797			ret = errno ? -errno : -ENODATA;
798			if (fclose(sysfsfp))
799				perror("_write_sysfs_string(): Failed to close dir");
800
801			goto error_free;
802		}
803
804		if (fclose(sysfsfp)) {
805			ret = -errno;
806			goto error_free;
807		}
808
809		if (strcmp(temp, val) != 0) {
810			fprintf(stderr,
811				"Possible failure in string write of %s "
812				"Should be %s written to %s/%s\n", temp, val,
813				basedir, filename);
814			ret = -1;
815		}
816	}
817
818error_free:
819	free(temp);
820
821	return ret;
822}
823
824/**
825 * write_sysfs_string_and_verify() - string write, readback and verify
826 * @filename: name of file to write to
827 * @basedir: the sysfs directory in which the file is to be found
828 * @val: the string to write
829 *
830 * Returns a value >= 0 on success, otherwise a negative error code.
831 **/
832int write_sysfs_string_and_verify(const char *filename, const char *basedir,
833				  const char *val)
834{
835	return _write_sysfs_string(filename, basedir, val, 1);
836}
837
838/**
839 * write_sysfs_string() - write string to a sysfs file
840 * @filename: name of file to write to
841 * @basedir: the sysfs directory in which the file is to be found
842 * @val: the string to write
843 *
844 * Returns a value >= 0 on success, otherwise a negative error code.
845 **/
846int write_sysfs_string(const char *filename, const char *basedir,
847		       const char *val)
848{
849	return _write_sysfs_string(filename, basedir, val, 0);
850}
851
852/**
853 * read_sysfs_posint() - read an integer value from file
854 * @filename: name of file to read from
855 * @basedir: the sysfs directory in which the file is to be found
856 *
857 * Returns the read integer value >= 0 on success, otherwise a negative error
858 * code.
859 **/
860int read_sysfs_posint(const char *filename, const char *basedir)
861{
862	int ret;
863	FILE  *sysfsfp;
864	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
865
866	if (!temp) {
867		fprintf(stderr, "Memory allocation failed");
868		return -ENOMEM;
869	}
870
871	ret = sprintf(temp, "%s/%s", basedir, filename);
872	if (ret < 0)
873		goto error_free;
874
875	sysfsfp = fopen(temp, "r");
876	if (!sysfsfp) {
877		ret = -errno;
878		goto error_free;
879	}
880
881	errno = 0;
882	if (fscanf(sysfsfp, "%d\n", &ret) != 1) {
883		ret = errno ? -errno : -ENODATA;
884		if (fclose(sysfsfp))
885			perror("read_sysfs_posint(): Failed to close dir");
886
887		goto error_free;
888	}
889
890	if (fclose(sysfsfp))
891		ret = -errno;
892
893error_free:
894	free(temp);
895
896	return ret;
897}
898
899/**
900 * read_sysfs_float() - read a float value from file
901 * @filename: name of file to read from
902 * @basedir: the sysfs directory in which the file is to be found
903 * @val: output the read float value
904 *
905 * Returns a value >= 0 on success, otherwise a negative error code.
906 **/
907int read_sysfs_float(const char *filename, const char *basedir, float *val)
908{
909	int ret = 0;
910	FILE  *sysfsfp;
911	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
912
913	if (!temp) {
914		fprintf(stderr, "Memory allocation failed");
915		return -ENOMEM;
916	}
917
918	ret = sprintf(temp, "%s/%s", basedir, filename);
919	if (ret < 0)
920		goto error_free;
921
922	sysfsfp = fopen(temp, "r");
923	if (!sysfsfp) {
924		ret = -errno;
925		goto error_free;
926	}
927
928	errno = 0;
929	if (fscanf(sysfsfp, "%f\n", val) != 1) {
930		ret = errno ? -errno : -ENODATA;
931		if (fclose(sysfsfp))
932			perror("read_sysfs_float(): Failed to close dir");
933
934		goto error_free;
935	}
936
937	if (fclose(sysfsfp))
938		ret = -errno;
939
940error_free:
941	free(temp);
942
943	return ret;
944}
945
946/**
947 * read_sysfs_string() - read a string from file
948 * @filename: name of file to read from
949 * @basedir: the sysfs directory in which the file is to be found
950 * @str: output the read string
951 *
952 * Returns a value >= 0 on success, otherwise a negative error code.
953 **/
954int read_sysfs_string(const char *filename, const char *basedir, char *str)
955{
956	int ret = 0;
957	FILE  *sysfsfp;
958	char *temp = malloc(strlen(basedir) + strlen(filename) + 2);
959
960	if (!temp) {
961		fprintf(stderr, "Memory allocation failed");
962		return -ENOMEM;
963	}
964
965	ret = sprintf(temp, "%s/%s", basedir, filename);
966	if (ret < 0)
967		goto error_free;
968
969	sysfsfp = fopen(temp, "r");
970	if (!sysfsfp) {
971		ret = -errno;
972		goto error_free;
973	}
974
975	errno = 0;
976	if (fscanf(sysfsfp, "%s\n", str) != 1) {
977		ret = errno ? -errno : -ENODATA;
978		if (fclose(sysfsfp))
979			perror("read_sysfs_string(): Failed to close dir");
980
981		goto error_free;
982	}
983
984	if (fclose(sysfsfp))
985		ret = -errno;
986
987error_free:
988	free(temp);
989
990	return ret;
991}