Example #1
0
/*
 * NAME:	record->comparecatkeys()
 * DESCRIPTION:	compare two (packed) catalog record keys
 */
int r_comparecatkeys(const CatKeyRec *key1, const CatKeyRec *key2)
{
  int diff;

  diff = key1->ckrParID - key2->ckrParID;
  if (diff)
    return diff;

  return d_relstring(key1->ckrCName, key2->ckrCName);
}
Example #2
0
/*
 * NAME:	compare_name()
 * DESCRIPTION:	compare reference list item with character name
 */
static
int compare_name(rsrcmap *map, const byte *ritem, const void *key)
{
  char name[256];
  const byte *nptr = map->nlist + d_getuw(ritem + 2);

  memcpy(name, nptr + 1, *nptr);
  name[*nptr] = 0;

  return d_relstring(name, key) == 0;
}
Example #3
0
File: hfs.c Project: 3a9LL/panda
/*
 * NAME:	hfs->getvol()
 * DESCRIPTION:	return a pointer to a mounted volume
 */
hfsvol *hfs_getvol(const char *name)
{
  hfsvol *vol;

  if (name == NULL)
    return curvol;

  for (vol = hfs_mounts; vol; vol = vol->next)
    {
      if (d_relstring(name, vol->mdb.drVN) == 0)
	return vol;
    }

  return NULL;
}
Example #4
0
/*
 * NAME:	vol->resolve()
 * DESCRIPTION:	translate a pathname; return catalog information
 */
int v_resolve(hfsvol **vol, const char *path,
              CatDataRec *data, long *parid, char *fname, node *np)
{
    long dirid;
    char name[HFS_MAX_FLEN + 1], *nptr;
    int found = 0;

    if (*path == 0)
        ERROR(ENOENT, "empty path");

    if (parid)
        *parid = 0;

    nptr = strchr(path, ':');

    if (*path == ':' || nptr == 0)
    {
        dirid = (*vol)->cwd;  /* relative path */

        if (*path == ':')
            ++path;

        if (*path == 0)
        {
            found = v_getdthread(*vol, dirid, data, 0);
            if (found == -1)
                goto fail;

            if (found)
            {
                if (parid)
                    *parid = data->u.dthd.thdParID;

                found = v_catsearch(*vol, data->u.dthd.thdParID,
                                    data->u.dthd.thdCName, data, fname, np);
                if (found == -1)
                    goto fail;
            }

            goto done;
        }
    }
    else
    {
        hfsvol *check;

        dirid = HFS_CNID_ROOTPAR;  /* absolute path */

        if (nptr - path > HFS_MAX_VLEN)
            ERROR(ENAMETOOLONG, 0);

        strncpy(name, path, nptr - path);
        name[nptr - path] = 0;

        for (check = hfs_mounts; check; check = check->next)
        {
            if (d_relstring(check->mdb.drVN, name) == 0)
            {
                *vol = check;
                break;
            }
        }
    }

    while (1)
    {
        while (*path == ':')
        {
            ++path;

            found = v_getdthread(*vol, dirid, data, 0);
            if (found == -1)
                goto fail;
            else if (! found)
                goto done;

            dirid = data->u.dthd.thdParID;
        }

        if (*path == 0)
        {
            found = v_getdthread(*vol, dirid, data, 0);
            if (found == -1)
                goto fail;

            if (found)
            {
                if (parid)
                    *parid = data->u.dthd.thdParID;

                found = v_catsearch(*vol, data->u.dthd.thdParID,
                                    data->u.dthd.thdCName, data, fname, np);
                if (found == -1)
                    goto fail;
            }

            goto done;
        }

        nptr = name;
        while (nptr < name + sizeof(name) - 1 && *path && *path != ':')
            *nptr++ = *path++;

        if (*path && *path != ':')
            ERROR(ENAMETOOLONG, 0);

        *nptr = 0;
        if (*path == ':')
            ++path;

        if (parid)
            *parid = dirid;

        found = v_catsearch(*vol, dirid, name, data, fname, np);
        if (found == -1)
            goto fail;

        if (! found)
        {
            if (*path && parid)
                *parid = 0;

            if (*path == 0 && fname)
                strcpy(fname, name);

            goto done;
        }

        switch (data->cdrType)
        {
        case cdrDirRec:
            if (*path == 0)
                goto done;

            dirid = data->u.dir.dirDirID;
            break;

        case cdrFilRec:
            if (*path == 0)
                goto done;

            ERROR(ENOTDIR, "invalid pathname");

        default:
            ERROR(EIO, "unexpected catalog record");
        }
    }

done:
    return found;

fail:
    return -1;
}
Example #5
0
/*
 * NAME:	hfs->rename()
 * DESCRIPTION:	change the name of and/or move a file or directory
 */
int hfs_rename(hfsvol *vol, const char *srcpath, const char *dstpath)
{
  hfsvol *srcvol;
  CatDataRec src, dst;
  unsigned long srcid, dstid;
  CatKeyRec key;
  char srcname[HFS_MAX_FLEN + 1], dstname[HFS_MAX_FLEN + 1];
  byte record[HFS_MAX_CATRECLEN];
  unsigned int reclen;
  int found, isdir, moving;
  node n;

  if (getvol(&vol) == -1 ||
      v_resolve(&vol, srcpath, &src, &srcid, srcname, 0) <= 0)
    goto fail;

  isdir  = (src.cdrType == cdrDirRec);
  srcvol = vol;

  found = v_resolve(&vol, dstpath, &dst, &dstid, dstname, 0);
  if (found == -1)
    goto fail;

  if (vol != srcvol)
    ERROR(EINVAL, "can't move across volumes");

  if (dstid == 0)
    ERROR(ENOENT, "bad destination path");

  if (found &&
      dst.cdrType == cdrDirRec &&
      dst.u.dir.dirDirID != src.u.dir.dirDirID)
    {
      dstid = dst.u.dir.dirDirID;
      strcpy(dstname, srcname);

      found = v_catsearch(vol, dstid, dstname, 0, 0, 0);
      if (found == -1)
	goto fail;
    }

  moving = (srcid != dstid);

  if (found)
    {
      const char *ptr;

      ptr = strrchr(dstpath, ':');
      if (ptr == 0)
	ptr = dstpath;
      else
	++ptr;

      if (*ptr)
	strcpy(dstname, ptr);

      if (! moving && strcmp(srcname, dstname) == 0)
	goto done;  /* source and destination are identical */

      if (moving || d_relstring(srcname, dstname))
	ERROR(EEXIST, "can't use destination name");
    }

  /* can't move anything into the root directory's parent */

  if (moving && dstid == HFS_CNID_ROOTPAR)
    ERROR(EINVAL, "can't move above root directory");

  if (moving && isdir)
    {
      unsigned long id;

      /* can't move root directory anywhere */

      if (src.u.dir.dirDirID == HFS_CNID_ROOTDIR)
	ERROR(EINVAL, "can't move root directory");

      /* make sure we aren't trying to move a directory inside itself */

      for (id = dstid; id != HFS_CNID_ROOTDIR; id = dst.u.dthd.thdParID)
	{
	  if (id == src.u.dir.dirDirID)
	    ERROR(EINVAL, "can't move directory inside itself");

	  if (v_getdthread(vol, id, &dst, 0) <= 0)
	    goto fail;
	}
    }

  if (vol->flags & HFS_VOL_READONLY)
    ERROR(EROFS, 0);

  /* change volume name */

  if (dstid == HFS_CNID_ROOTPAR)
    {
      if (! validvname(dstname))
	goto fail;

      strcpy(vol->mdb.drVN, dstname);
      vol->flags |= HFS_VOL_UPDATE_MDB;
    }

  /* remove source record */

  r_makecatkey(&key, srcid, srcname);
  r_packcatkey(&key, record, 0);

  if (bt_delete(&vol->cat, record) == -1)
    goto fail;

  /* insert destination record */

  r_makecatkey(&key, dstid, dstname);
  r_packcatrec(&key, &src, record, &reclen);

  if (bt_insert(&vol->cat, record, reclen) == -1)
    goto fail;

  /* update thread record */

  if (isdir)
    {
      if (v_getdthread(vol, src.u.dir.dirDirID, &dst, &n) <= 0)
	goto fail;

      dst.u.dthd.thdParID = dstid;
      strcpy(dst.u.dthd.thdCName, dstname);

      if (v_putcatrec(&dst, &n) == -1)
	goto fail;
    }
  else
    {
      found = v_getfthread(vol, src.u.fil.filFlNum, &dst, &n);
      if (found == -1)
	goto fail;

      if (found)
	{
	  dst.u.fthd.fthdParID = dstid;
	  strcpy(dst.u.fthd.fthdCName, dstname);

	  if (v_putcatrec(&dst, &n) == -1)
	    goto fail;
	}
    }

  /* update directory valences */

  if (moving)
    {
      if (v_adjvalence(vol, srcid, isdir, -1) == -1 ||
	  v_adjvalence(vol, dstid, isdir,  1) == -1)
	goto fail;
    }

done:
  return 0;

fail:
  return -1;
}