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;
}

Reply via email to