xref: /minix3/minix/lib/libvtreefs/link.c (revision 7c48de6cc4c6d56f2277d378dba01dbac8a8c3b9)
15eefd0feSDavid van Moolenbroek /* VTreeFS - link.c - support for symbolic links and device nodes */
2433d6423SLionel Sambuc 
3433d6423SLionel Sambuc #include "inc.h"
4433d6423SLionel Sambuc 
5693ad767SDavid van Moolenbroek /*
65eefd0feSDavid van Moolenbroek  * Retrieve a symbolic link target.
7433d6423SLionel Sambuc  */
8693ad767SDavid van Moolenbroek ssize_t
fs_rdlink(ino_t ino_nr,struct fsdriver_data * data,size_t bytes)9693ad767SDavid van Moolenbroek fs_rdlink(ino_t ino_nr, struct fsdriver_data * data, size_t bytes)
10693ad767SDavid van Moolenbroek {
11433d6423SLionel Sambuc 	char path[PATH_MAX];
12433d6423SLionel Sambuc 	struct inode *node;
13433d6423SLionel Sambuc 	size_t len;
14433d6423SLionel Sambuc 	int r;
15433d6423SLionel Sambuc 
160dc5c83eSDavid van Moolenbroek 	if ((node = find_inode(ino_nr)) == NULL)
17433d6423SLionel Sambuc 		return EINVAL;
18433d6423SLionel Sambuc 
195eefd0feSDavid van Moolenbroek 	/* The hook should be provided for any FS that adds symlink inodes.. */
205eefd0feSDavid van Moolenbroek 	if (vtreefs_hooks->rdlink_hook == NULL)
215eefd0feSDavid van Moolenbroek 		return ENOSYS;
225eefd0feSDavid van Moolenbroek 
23433d6423SLionel Sambuc 	assert(!is_inode_deleted(node));	/* symlinks cannot be opened */
24433d6423SLionel Sambuc 
25433d6423SLionel Sambuc 	r = vtreefs_hooks->rdlink_hook(node, path, sizeof(path),
26433d6423SLionel Sambuc 	    get_inode_cbdata(node));
27433d6423SLionel Sambuc 	if (r != OK) return r;
28433d6423SLionel Sambuc 
29433d6423SLionel Sambuc 	len = strlen(path);
30433d6423SLionel Sambuc 	assert(len > 0 && len < sizeof(path));
31433d6423SLionel Sambuc 
320dc5c83eSDavid van Moolenbroek 	if (len > bytes)
330dc5c83eSDavid van Moolenbroek 		len = bytes;
34433d6423SLionel Sambuc 
35433d6423SLionel Sambuc 	/* Copy out the result. */
360dc5c83eSDavid van Moolenbroek 	if ((r = fsdriver_copyout(data, 0, path, len)) != OK)
370dc5c83eSDavid van Moolenbroek 		return r;
38433d6423SLionel Sambuc 
390dc5c83eSDavid van Moolenbroek 	return len;
40433d6423SLionel Sambuc }
415eefd0feSDavid van Moolenbroek 
425eefd0feSDavid van Moolenbroek /*
435eefd0feSDavid van Moolenbroek  * Create a symbolic link.
445eefd0feSDavid van Moolenbroek  */
455eefd0feSDavid van Moolenbroek int
fs_slink(ino_t dir_nr,char * name,uid_t uid,gid_t gid,struct fsdriver_data * data,size_t bytes)465eefd0feSDavid van Moolenbroek fs_slink(ino_t dir_nr, char * name, uid_t uid, gid_t gid,
475eefd0feSDavid van Moolenbroek 	struct fsdriver_data * data, size_t bytes)
485eefd0feSDavid van Moolenbroek {
495eefd0feSDavid van Moolenbroek 	char path[PATH_MAX];
505eefd0feSDavid van Moolenbroek 	struct inode *node;
51*7c48de6cSDavid van Moolenbroek 	struct inode_stat istat;
525eefd0feSDavid van Moolenbroek 	int r;
535eefd0feSDavid van Moolenbroek 
545eefd0feSDavid van Moolenbroek 	if ((node = find_inode(dir_nr)) == NULL)
555eefd0feSDavid van Moolenbroek 		return EINVAL;
565eefd0feSDavid van Moolenbroek 
575eefd0feSDavid van Moolenbroek 	if (vtreefs_hooks->slink_hook == NULL)
585eefd0feSDavid van Moolenbroek 		return ENOSYS;
595eefd0feSDavid van Moolenbroek 
605eefd0feSDavid van Moolenbroek 	if (get_inode_by_name(node, name) != NULL)
615eefd0feSDavid van Moolenbroek 		return EEXIST;
625eefd0feSDavid van Moolenbroek 
635eefd0feSDavid van Moolenbroek 	if (bytes >= sizeof(path))
645eefd0feSDavid van Moolenbroek 		return ENAMETOOLONG;
655eefd0feSDavid van Moolenbroek 
665eefd0feSDavid van Moolenbroek 	if ((r = fsdriver_copyin(data, 0, path, bytes)) != OK)
675eefd0feSDavid van Moolenbroek 		return r;
685eefd0feSDavid van Moolenbroek 	path[bytes] = 0;
695eefd0feSDavid van Moolenbroek 
70*7c48de6cSDavid van Moolenbroek 	memset(&istat, 0, sizeof(istat));
71*7c48de6cSDavid van Moolenbroek 	istat.mode = S_IFLNK | RWX_MODES;
72*7c48de6cSDavid van Moolenbroek 	istat.uid = uid;
73*7c48de6cSDavid van Moolenbroek 	istat.gid = gid;
74*7c48de6cSDavid van Moolenbroek 	istat.size = strlen(path);
75*7c48de6cSDavid van Moolenbroek 	istat.dev = 0;
765eefd0feSDavid van Moolenbroek 
77*7c48de6cSDavid van Moolenbroek 	return vtreefs_hooks->slink_hook(node, name, &istat, path,
785eefd0feSDavid van Moolenbroek 	    get_inode_cbdata(node));
795eefd0feSDavid van Moolenbroek }
805eefd0feSDavid van Moolenbroek 
815eefd0feSDavid van Moolenbroek /*
825eefd0feSDavid van Moolenbroek  * Create a device node.
835eefd0feSDavid van Moolenbroek  */
845eefd0feSDavid van Moolenbroek int
fs_mknod(ino_t dir_nr,char * name,mode_t mode,uid_t uid,gid_t gid,dev_t rdev)855eefd0feSDavid van Moolenbroek fs_mknod(ino_t dir_nr, char * name, mode_t mode, uid_t uid, gid_t gid,
865eefd0feSDavid van Moolenbroek 	dev_t rdev)
875eefd0feSDavid van Moolenbroek {
885eefd0feSDavid van Moolenbroek 	struct inode *node;
89*7c48de6cSDavid van Moolenbroek 	struct inode_stat istat;
905eefd0feSDavid van Moolenbroek 
915eefd0feSDavid van Moolenbroek 	if ((node = find_inode(dir_nr)) == NULL)
925eefd0feSDavid van Moolenbroek 		return EINVAL;
935eefd0feSDavid van Moolenbroek 
945eefd0feSDavid van Moolenbroek 	if (get_inode_by_name(node, name) != NULL)
955eefd0feSDavid van Moolenbroek 		return EEXIST;
965eefd0feSDavid van Moolenbroek 
975eefd0feSDavid van Moolenbroek 	if (vtreefs_hooks->mknod_hook == NULL)
985eefd0feSDavid van Moolenbroek 		return ENOSYS;
995eefd0feSDavid van Moolenbroek 
100*7c48de6cSDavid van Moolenbroek 	memset(&istat, 0, sizeof(istat));
101*7c48de6cSDavid van Moolenbroek 	istat.mode = mode;
102*7c48de6cSDavid van Moolenbroek 	istat.uid = uid;
103*7c48de6cSDavid van Moolenbroek 	istat.gid = gid;
104*7c48de6cSDavid van Moolenbroek 	istat.size = 0;
105*7c48de6cSDavid van Moolenbroek 	istat.dev = rdev;
1065eefd0feSDavid van Moolenbroek 
107*7c48de6cSDavid van Moolenbroek 	return vtreefs_hooks->mknod_hook(node, name, &istat,
1085eefd0feSDavid van Moolenbroek 	    get_inode_cbdata(node));
1095eefd0feSDavid van Moolenbroek }
1105eefd0feSDavid van Moolenbroek 
1115eefd0feSDavid van Moolenbroek /*
1125eefd0feSDavid van Moolenbroek  * Unlink a node.
1135eefd0feSDavid van Moolenbroek  */
1145eefd0feSDavid van Moolenbroek int
fs_unlink(ino_t dir_nr,char * name,int __unused call)1155eefd0feSDavid van Moolenbroek fs_unlink(ino_t dir_nr, char * name, int __unused call)
1165eefd0feSDavid van Moolenbroek {
1175eefd0feSDavid van Moolenbroek 	struct inode *dir_node, *node;
1185eefd0feSDavid van Moolenbroek 
1195eefd0feSDavid van Moolenbroek 	if ((dir_node = find_inode(dir_nr)) == NULL)
1205eefd0feSDavid van Moolenbroek 		return EINVAL;
1215eefd0feSDavid van Moolenbroek 
1225eefd0feSDavid van Moolenbroek 	if ((node = get_inode_by_name(dir_node, name)) == NULL)
1235eefd0feSDavid van Moolenbroek 		return ENOENT;
1245eefd0feSDavid van Moolenbroek 
1255eefd0feSDavid van Moolenbroek 	if (vtreefs_hooks->unlink_hook == NULL)
1265eefd0feSDavid van Moolenbroek 		return ENOSYS;
1275eefd0feSDavid van Moolenbroek 
1285eefd0feSDavid van Moolenbroek 	return vtreefs_hooks->unlink_hook(node, get_inode_cbdata(node));
1295eefd0feSDavid van Moolenbroek }
130