/* ls2.c * purpose list contents of directory or directories * action if no args, use . else list files in args * note uses stat and pwd.h and grp.h * BUG: try ls2 /tmp * NOTE: this one skips the dot files, because that's easy to do */ #include #include #include #include #include #include #define IDNUM_LEN 20 /* length of UID/GID int */ #define DATE_FMT "%b %e %H:%M" /* text format: mon dd hh:mm */ #ifdef NUMERIC_DATE #define DATE_FMT "%Y-%m-%d %H:%M" /* numeric: yyyy-mm-dd hh:mm */ #endif void do_ls(char[]); void dostat(char *); void show_file_info( char *, struct stat *); char *mode_to_letters( int , char [] ); char *uid_to_name( uid_t ); char *gid_to_name( gid_t ); char *fmt_time(time_t); int main(int ac, char *av[]) { if ( ac == 1 ) do_ls( "." ); else while ( --ac ){ printf("%s:\n", *++av ); do_ls( *av ); } } void do_ls( char dirname[] ) /* * list files in directory called dirname */ { DIR *dir_ptr; /* the directory */ struct dirent *direntp; /* each entry */ // a directory is file containing a list of (inum,name) pairs if ( ( dir_ptr = opendir( dirname ) ) == NULL ) fprintf(stderr,"ls2: cannot open %s\n", dirname); else { while ( ( direntp = readdir( dir_ptr ) ) != NULL ) if ( direntp->d_name[0] != '.' ) dostat( direntp->d_name ); closedir(dir_ptr); } } void dostat( char *filename ) { struct stat info; // remember wehre we are if we are going to chdir if ( stat(filename, &info) == -1 ) /* cannot stat */ perror( filename ); /* say why */ else /* else show info */ show_file_info( filename, &info ); } void show_file_info( char *filename, struct stat *info_p ) /* * display the info about 'filename'. The info is stored in struct at *info_p */ { char modestr[11]; mode_to_letters( info_p->st_mode, modestr ); printf( "%s " , modestr ); printf( "%d " , (int) info_p->st_nlink); printf( "%-6s " , uid_to_name(info_p->st_uid) ); printf( "%-8s " , gid_to_name(info_p->st_gid) ); printf( "%5ld " , (long)info_p->st_size); printf( "%s " , fmt_time( info_p->st_mtime ) ); printf( "%s\n" , filename ); } /* * utility functions */ #define MAXDATELEN 100 char * fmt_time( time_t timeval ) /* * formats time for human consumption. * Uses localtime to convert the timeval into a struct of elements * (see localtime(3)) and uses strftime to format the data * the format can be numeric format, text format for recent, text fmt for > 6months */ { static char result[MAXDATELEN]; char *fmt = DATE_FMT; struct tm *tp = localtime(&timeval); /* convert time */ strftime(result, MAXDATELEN, fmt, tp); /* format it */ return result; } /* * This function takes a mode value and a char array * and puts into the char array the file type and the * nine letters that correspond to the bits in mode. * NOTE: It does not code setuid, setgid, and sticky * codes */ char *mode_to_letters( int mode, char str[] ) { strcpy( str, "----------" ); /* default=no perms */ if ( S_ISDIR(mode) ) str[0] = 'd'; /* directory? */ if ( S_ISCHR(mode) ) str[0] = 'c'; /* char devices */ if ( S_ISBLK(mode) ) str[0] = 'b'; /* block device */ if ( mode & S_IRUSR ) str[1] = 'r'; /* 3 bits for user */ if ( mode & S_IWUSR ) str[2] = 'w'; if ( mode & S_IXUSR ) str[3] = 'x'; if ( mode & S_IRGRP ) str[4] = 'r'; /* 3 bits for group */ if ( mode & S_IWGRP ) str[5] = 'w'; if ( mode & S_IXGRP ) str[6] = 'x'; if ( mode & S_IROTH ) str[7] = 'r'; /* 3 bits for other */ if ( mode & S_IWOTH ) str[8] = 'w'; if ( mode & S_IXOTH ) str[9] = 'x'; return str; } #include char *uid_to_name( uid_t uid ) /* * returns pointer to username associated with uid, uses getpw() */ { struct passwd *pw_ptr; static char numstr[IDNUM_LEN]; if ( ( pw_ptr = getpwuid( uid ) ) == NULL ){ sprintf(numstr,"%d", uid); return numstr; } else return pw_ptr->pw_name ; } #include char *gid_to_name( gid_t gid ) /* * returns pointer to group number gid. used getgrgid(3) */ { struct group *grp_ptr; static char numstr[IDNUM_LEN]; if ( ( grp_ptr = getgrgid(gid) ) == NULL ){ sprintf(numstr,"%d", gid); return numstr; } else return grp_ptr->gr_name; }