I wasn't able to get this working today, but I believe I made good progress. Unfortunately, I'm traveling all day tomorrow and will have questionable network access while away for about a week.
I've attached my version of _p9dir.c with the OS X code in it. The addition is pretty simple, and I tested a dummy program with nearly the same code and it prints the real size in bytes, but something isn't right. I rebuilt and installed lib9 and ls, but ls still doesn't show anything useful. It may just be that I have to rebuild something more than I have. Forwarded in the hope that someone who's looked at p9p's guts more than I can tell me what that might be. Anthony
#include <u.h> #define NOPLAN9DEFINES #include <libc.h> #include <sys/types.h> #include <sys/stat.h> #include <dirent.h> #include <pwd.h> #include <grp.h> #if defined(__FreeBSD__) #include <sys/disk.h> #include <sys/disklabel.h> #include <sys/ioctl.h> #endif #if defined(__OpenBSD__) #include <sys/disklabel.h> #include <sys/ioctl.h> #define _HAVEDISKLABEL static int diskdev[] = { 151, /* aacd */ 116, /* ad */ 157, /* ar */ 118, /* afd */ 133, /* amrd */ 13, /* da */ 102, /* fla */ 109, /* idad */ 95, /* md */ 131, /* mlxd */ 168, /* pst */ 147, /* twed */ 43, /* vn */ 3, /* wd */ 87, /* wfd */ 4, /* da on FreeBSD 5 */ }; static int isdisk(struct stat *st) { int i, dev; if(!S_ISCHR(st->st_mode)) return 0; dev = major(st->st_rdev); for(i=0; i<nelem(diskdev); i++) if(diskdev[i] == dev) return 1; return 0; } #endif #if defined(__FreeBSD__) /* maybe OpenBSD too? */ char *diskdev[] = { "aacd", "ad", "ar", "afd", "amrd", "da", "fla", "idad", "md", "mlxd", "pst", "twed", "vn", "wd", "wfd", "da", }; static int isdisk(struct stat *st) { char *name; int i, len; if(!S_ISCHR(st->st_mode)) return 0; name = devname(st->st_rdev, S_IFCHR); for(i=0; i<nelem(diskdev); i++){ len = strlen(diskdev[i]); if(strncmp(diskdev[i], name, len) == 0 && isdigit((uchar)name[len])) return 1; } return 0; } #endif #if defined(__linux__) #include <linux/hdreg.h> #include <linux/fs.h> #include <sys/ioctl.h> #undef major #define major(dev) ((int)(((dev) >> 8) & 0xff)) static vlong disksize(int fd, int dev) { u64int u64; long l; struct hd_geometry geo; memset(&geo, 0, sizeof geo); l = 0; u64 = 0; #ifdef BLKGETSIZE64 if(ioctl(fd, BLKGETSIZE64, &u64) >= 0) return u64; #endif if(ioctl(fd, BLKGETSIZE, &l) >= 0) return l*512; if(ioctl(fd, HDIO_GETGEO, &geo) >= 0) return (vlong)geo.heads*geo.sectors*geo.cylinders*512; return 0; } #define _HAVEDISKSIZE #endif #if defined(__APPLE__) #include <sys/disk.h> static vlong disksize(int fd, int dev) { u64int blockCount; u32int blockSize; ioctl(fd, DKIOCGETBLOCKSIZE, &blockSize); ioctl(fd, DKIOCGETBLOCKCOUNT, &blockCount); if(blockSize >0 && blockCount > 0) return blockCount*blockSize; return 0; } #define _HAVEDISKSIZE #endif #if !defined(__linux__) && !defined(__sun__) #define _HAVESTGEN #endif int _p9usepwlibrary = 1; /* * Caching the last group and passwd looked up is * a significant win (stupidly enough) on most systems. * It's not safe for threaded programs, but neither is using * getpwnam in the first place, so I'm not too worried. */ int _p9dir(struct stat *lst, struct stat *st, char *name, Dir *d, char **str, char *estr) { char *s; char tmp[20]; static struct group *g; static struct passwd *p; static int gid, uid; int sz, fd; fd = -1; USED(fd); sz = 0; if(d) memset(d, 0, sizeof *d); /* name */ s = strrchr(name, '/'); if(s) s++; if(!s || !*s) s = name; if(*s == '/') s++; if(*s == 0) s = "/"; if(d){ if(*str + strlen(s)+1 > estr) d->name = "oops"; else{ strcpy(*str, s); d->name = *str; *str += strlen(*str)+1; } } sz += strlen(s)+1; /* user */ if(p && st->st_uid == uid && p->pw_uid == uid) ; else if(_p9usepwlibrary){ p = getpwuid(st->st_uid); uid = st->st_uid; } if(p == nil || st->st_uid != uid || p->pw_uid != uid){ snprint(tmp, sizeof tmp, "%d", (int)st->st_uid); s = tmp; }else s = p->pw_name; sz += strlen(s)+1; if(d){ if(*str+strlen(s)+1 > estr) d->uid = "oops"; else{ strcpy(*str, s); d->uid = *str; *str += strlen(*str)+1; } } /* group */ if(g && st->st_gid == gid && g->gr_gid == gid) ; else if(_p9usepwlibrary){ g = getgrgid(st->st_gid); gid = st->st_gid; } if(g == nil || st->st_gid != gid || g->gr_gid != gid){ snprint(tmp, sizeof tmp, "%d", (int)st->st_gid); s = tmp; }else s = g->gr_name; sz += strlen(s)+1; if(d){ if(*str + strlen(s)+1 > estr) d->gid = "oops"; else{ strcpy(*str, s); d->gid = *str; *str += strlen(*str)+1; } } if(d){ d->type = 'M'; d->muid = ""; d->qid.path = ((uvlong)st->st_dev<<32) | st->st_ino; #ifdef _HAVESTGEN d->qid.vers = st->st_gen; #endif if(d->qid.vers == 0) d->qid.vers = st->st_mtime + st->st_ctime; d->mode = st->st_mode&0777; d->atime = st->st_atime; d->mtime = st->st_mtime; d->length = st->st_size; if(S_ISDIR(st->st_mode)){ d->length = 0; d->mode |= DMDIR; d->qid.type = QTDIR; } if(S_ISLNK(lst->st_mode)) /* yes, lst not st */ d->mode |= DMSYMLINK; if(S_ISFIFO(st->st_mode)) d->mode |= DMNAMEDPIPE; if(S_ISSOCK(st->st_mode)) d->mode |= DMSOCKET; if(S_ISBLK(st->st_mode)){ d->mode |= DMDEVICE; d->qid.path = ('b'<<16)|st->st_rdev; } if(S_ISCHR(st->st_mode)){ d->mode |= DMDEVICE; d->qid.path = ('c'<<16)|st->st_rdev; } /* fetch real size for disks */ #ifdef _HAVEDISKSIZE if(S_ISBLK(st->st_mode) && (fd = open(name, O_RDONLY)) >= 0){ d->length = disksize(fd, major(st->st_dev)); close(fd); } #endif #if defined(DIOCGMEDIASIZE) if(isdisk(st)){ int fd; off_t mediasize; if((fd = open(name, O_RDONLY)) >= 0){ if(ioctl(fd, DIOCGMEDIASIZE, &mediasize) >= 0) d->length = mediasize; close(fd); } } #elif defined(_HAVEDISKLABEL) if(isdisk(st)){ int fd, n; struct disklabel lab; if((fd = open(name, O_RDONLY)) < 0) goto nosize; if(ioctl(fd, DIOCGDINFO, &lab) < 0) goto nosize; n = minor(st->st_rdev)&7; if(n >= lab.d_npartitions) goto nosize; d->length = (vlong)(lab.d_partitions[n].p_size) * lab.d_secsize; nosize: if(fd >= 0) close(fd); } #endif } return sz; }