| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* | 
 | 2 |  *  linux/fs/affs/symlink.c | 
 | 3 |  * | 
 | 4 |  *  1995  Hans-Joachim Widmaier - Modified for affs. | 
 | 5 |  * | 
 | 6 |  *  Copyright (C) 1991, 1992  Linus Torvalds | 
 | 7 |  * | 
 | 8 |  *  affs symlink handling code | 
 | 9 |  */ | 
 | 10 |  | 
 | 11 | #include "affs.h" | 
 | 12 |  | 
 | 13 | static int affs_symlink_readpage(struct file *file, struct page *page) | 
 | 14 | { | 
 | 15 | 	struct buffer_head *bh; | 
 | 16 | 	struct inode *inode = page->mapping->host; | 
 | 17 | 	char *link = kmap(page); | 
 | 18 | 	struct slink_front *lf; | 
 | 19 | 	int err; | 
 | 20 | 	int			 i, j; | 
 | 21 | 	char			 c; | 
 | 22 | 	char			 lc; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 23 |  | 
 | 24 | 	pr_debug("AFFS: follow_link(ino=%lu)\n",inode->i_ino); | 
 | 25 |  | 
 | 26 | 	err = -EIO; | 
 | 27 | 	bh = affs_bread(inode->i_sb, inode->i_ino); | 
 | 28 | 	if (!bh) | 
 | 29 | 		goto fail; | 
 | 30 | 	i  = 0; | 
 | 31 | 	j  = 0; | 
 | 32 | 	lf = (struct slink_front *)bh->b_data; | 
 | 33 | 	lc = 0; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 34 |  | 
 | 35 | 	if (strchr(lf->symname,':')) {	/* Handle assign or volume name */ | 
| Al Viro | 2933392 | 2010-01-24 00:04:07 -0500 | [diff] [blame] | 36 | 		struct affs_sb_info *sbi = AFFS_SB(inode->i_sb); | 
 | 37 | 		char *pf; | 
 | 38 | 		spin_lock(&sbi->symlink_lock); | 
 | 39 | 		pf = sbi->s_prefix ? sbi->s_prefix : "/"; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 40 | 		while (i < 1023 && (c = pf[i])) | 
 | 41 | 			link[i++] = c; | 
| Al Viro | 2933392 | 2010-01-24 00:04:07 -0500 | [diff] [blame] | 42 | 		spin_unlock(&sbi->symlink_lock); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 43 | 		while (i < 1023 && lf->symname[j] != ':') | 
 | 44 | 			link[i++] = lf->symname[j++]; | 
 | 45 | 		if (i < 1023) | 
 | 46 | 			link[i++] = '/'; | 
 | 47 | 		j++; | 
 | 48 | 		lc = '/'; | 
 | 49 | 	} | 
 | 50 | 	while (i < 1023 && (c = lf->symname[j])) { | 
 | 51 | 		if (c == '/' && lc == '/' && i < 1020) {	/* parent dir */ | 
 | 52 | 			link[i++] = '.'; | 
 | 53 | 			link[i++] = '.'; | 
 | 54 | 		} | 
 | 55 | 		link[i++] = c; | 
 | 56 | 		lc = c; | 
 | 57 | 		j++; | 
 | 58 | 	} | 
 | 59 | 	link[i] = '\0'; | 
 | 60 | 	affs_brelse(bh); | 
 | 61 | 	SetPageUptodate(page); | 
 | 62 | 	kunmap(page); | 
 | 63 | 	unlock_page(page); | 
 | 64 | 	return 0; | 
 | 65 | fail: | 
 | 66 | 	SetPageError(page); | 
 | 67 | 	kunmap(page); | 
 | 68 | 	unlock_page(page); | 
 | 69 | 	return err; | 
 | 70 | } | 
 | 71 |  | 
| Christoph Hellwig | f5e54d6 | 2006-06-28 04:26:44 -0700 | [diff] [blame] | 72 | const struct address_space_operations affs_symlink_aops = { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 73 | 	.readpage	= affs_symlink_readpage, | 
 | 74 | }; | 
 | 75 |  | 
| Arjan van de Ven | 754661f | 2007-02-12 00:55:38 -0800 | [diff] [blame] | 76 | const struct inode_operations affs_symlink_inode_operations = { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 77 | 	.readlink	= generic_readlink, | 
 | 78 | 	.follow_link	= page_follow_link_light, | 
 | 79 | 	.put_link	= page_put_link, | 
 | 80 | 	.setattr	= affs_notify_change, | 
 | 81 | }; |