nuttx-apps/examples/stat/stat_main.c

350 lines
9.1 KiB
C

/****************************************************************************
* examples/stat/stat_main.c
*
* Copyright (C) 2008, 2017 Gregory Nutt. All rights reserved.
* Author: Gregory Nutt <gnutt@nuttx.org>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* 3. Neither the name NuttX nor the names of its contributors may be
* used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <sys/stat.h>
#include <sys/statfs.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <errno.h>
/****************************************************************************
* Private Data
****************************************************************************/
static struct mallinfo g_mmbefore;
static struct mallinfo g_mmprevious;
static struct mallinfo g_mmafter;
/****************************************************************************
* Private Functions
****************************************************************************/
static void showusage(FAR struct mallinfo *mmbefore,
FAR struct mallinfo *mmafter, FAR const char *msg)
{
if (mmbefore->uordblks != mmafter->uordblks)
{
printf("\n%s:\n", msg);
printf("VARIABLE BEFORE AFTER\n");
printf("======== ======== ========\n");
printf("arena %8x %8x\n", mmbefore->arena, mmafter->arena);
printf("ordblks %8d %8d\n", mmbefore->ordblks, mmafter->ordblks);
printf("mxordblk %8x %8x\n", mmbefore->mxordblk, mmafter->mxordblk);
printf("uordblks %8x %8x\n", mmbefore->uordblks, mmafter->uordblks);
printf("fordblks %8x %8x\n", mmbefore->fordblks, mmafter->fordblks);
}
}
static void stepusage(void)
{
/* Get the current memory usage */
#ifdef CONFIG_CAN_PASS_STRUCTS
g_mmafter = mallinfo();
#else
(void)mallinfo(&g_mmafter);
#endif
/* Show the change from the previous loop */
showusage(&g_mmprevious, &g_mmafter, "Step memory leak");
/* Set up for the next test */
#ifdef CONFIG_CAN_PASS_STRUCTS
g_mmprevious = g_mmafter;
#else
memcpy(&g_mmprevious, &g_mmafter, sizeof(struct mallinfo));
#endif
}
static void endusage(void)
{
#ifdef CONFIG_CAN_PASS_STRUCTS
g_mmafter = mallinfo();
#else
(void)mallinfo(&g_mmafter);
#endif
showusage(&g_mmbefore, &g_mmafter, "End-of-test memory leak");
}
static void dump_stat(FAR struct stat *buf)
{
char details[] = "----------";
if (S_ISLNK(buf->st_mode))
{
details[0] = 'l'; /* Takes precedence over type of the target */
}
else if (S_ISBLK(buf->st_mode))
{
details[0] = 'b';
}
else if (S_ISCHR(buf->st_mode))
{
details[0] = 'c';
}
else if (S_ISDIR(buf->st_mode))
{
details[0] = 'd';
}
else if (S_ISMTD(buf->st_mode))
{
details[0] = 'f';
}
else if (S_ISSHM(buf->st_mode))
{
details[0] = 'h';
}
else if (S_ISMQ(buf->st_mode))
{
details[0] = 'm';
}
else if (S_ISSOCK(buf->st_mode))
{
details[0] = 'n';
}
else if (S_ISSEM(buf->st_mode))
{
details[0] = 's';
}
else if (!S_ISREG(buf->st_mode))
{
details[0] = '?';
}
if ((buf->st_mode & S_IRUSR) != 0)
{
details[1]='r';
}
if ((buf->st_mode & S_IWUSR) != 0)
{
details[2]='w';
}
if ((buf->st_mode & S_IXUSR) != 0)
{
details[3]='x';
}
if ((buf->st_mode & S_IRGRP) != 0)
{
details[4]='r';
}
if ((buf->st_mode & S_IWGRP) != 0)
{
details[5]='w';
}
if ((buf->st_mode & S_IXGRP) != 0)
{
details[6]='x';
}
if ((buf->st_mode & S_IROTH) != 0)
{
details[7]='r';
}
if ((buf->st_mode & S_IWOTH) != 0)
{
details[8]='w';
}
if ((buf->st_mode & S_IXOTH) != 0)
{
details[9]='x';
}
printf("stat buffer:\n");
printf(" st_mode: %04x %s\n", buf->st_mode, details);
printf(" st_size: %llu\n", (unsigned long long)buf->st_size);
printf(" st_blksize: %lu\n", (unsigned long)buf->st_blksize);
printf(" st_blocks: %lu\n", (unsigned long)buf->st_blocks);
printf(" st_atime: %08lx\n", (unsigned long)buf->st_atime);
printf(" st_mtime: %08lx\n", (unsigned long)buf->st_mtime);
printf(" st_ctime: %08lx\n", (unsigned long)buf->st_ctime);
}
static void dump_statfs(FAR struct statfs *buf)
{
printf("statfs buffer:\n");
printf(" f_type: %lu\n", (unsigned long)buf->f_type);
printf(" f_namelen: %lu\n", (unsigned long)buf->f_namelen);
printf(" f_bsize: %lu\n", (unsigned long)buf->f_bsize);
printf(" f_blocks: %llu\n", (unsigned long long)buf->f_blocks);
printf(" f_bfree: %llu\n", (unsigned long long)buf->f_bfree);
printf(" f_bavail: %llu\n", (unsigned long long)buf->f_bavail);
printf(" f_files: %llu\n", (unsigned long long)buf->f_files);
printf(" f_ffree: %llu\n", (unsigned long long)buf->f_ffree);
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* stat_main
****************************************************************************/
int main(int argc, FAR char *argv[])
{
FAR const char *path;
struct stat statbuf;
struct statfs statfsbuf;
bool isreg;
int ret;
/* Argument is expected... the path to the file to test */
if (argc != 2)
{
fprintf(stderr,
"ERROR: Invalid number of arguments: %d (expected 2)\n",
argc);
return EXIT_FAILURE;
}
path = argv[1];
printf("Testing path: \"%s\"\n", path);
/* Set up memory monitoring */
#ifdef CONFIG_CAN_PASS_STRUCTS
g_mmbefore = mallinfo();
g_mmprevious = g_mmbefore;
#else
(void)mallinfo(&g_mmbefore);
memcpy(&g_mmprevious, &g_mmbefore, sizeof(struct mallinfo));
#endif
/* Try stat first */
printf("\nTest stat(%s)\n", path);
ret = stat(path, &statbuf);
if (ret < 0)
{
int errcode = errno;
fprintf(stderr,
"ERROR: stat(%s) failed: %d\n",
path, errcode);
return EXIT_FAILURE;
}
dump_stat(&statbuf);
stepusage();
isreg = S_ISREG(statbuf.st_mode);
/* Try statfs */
printf("\nTest statfs(%s)\n", path);
ret = statfs(path, &statfsbuf);
if (ret < 0)
{
int errcode = errno;
fprintf(stderr,
"ERROR: statfs(%s) failed: %d\n",
path, errcode);
}
dump_statfs(&statfsbuf);
stepusage();
/* Try fstat (only if it is a regular file) */
if (isreg)
{
int fd;
printf("\nOpen(%s)\n", path);
fd = open(path, O_RDONLY);
if (fd < 0)
{
int errcode = errno;
fprintf(stderr,
"ERROR: open(%s) failed: %d\n",
path, errcode);
return EXIT_FAILURE;
}
/* Try fstat */
printf("\nTest fstat(%s)\n", path);
ret = fstat(fd, &statbuf);
if (ret < 0)
{
int errcode = errno;
fprintf(stderr,
"ERROR: fstat(%s) failed: %d\n",
path, errcode);
}
else
{
dump_stat(&statbuf);
}
/* Try fstatfs */
printf("\nTest fstatfs(%s)\n", path);
ret = fstatfs(fd, &statfsbuf);
if (ret < 0)
{
int errcode = errno;
fprintf(stderr,
"ERROR: fstatfs(%s) failed: %d\n",
path, errcode);
}
else
{
dump_statfs(&statfsbuf);
}
close(fd);
stepusage();
}
endusage();
return 0;
}