blob: ea5b69a18ba9ce9165d8ea083bfffbb40e13a600 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
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
13static 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;
Linus Torvalds1da177e2005-04-16 15:20:36 -070019 int i, j;
20 char c;
21 char lc;
Linus Torvalds1da177e2005-04-16 15:20:36 -070022
Fabian Frederick9606d9a2014-06-06 14:37:25 -070023 pr_debug("follow_link(ino=%lu)\n", inode->i_ino);
Linus Torvalds1da177e2005-04-16 15:20:36 -070024
Linus Torvalds1da177e2005-04-16 15:20:36 -070025 bh = affs_bread(inode->i_sb, inode->i_ino);
26 if (!bh)
27 goto fail;
28 i = 0;
29 j = 0;
30 lf = (struct slink_front *)bh->b_data;
31 lc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -070032
33 if (strchr(lf->symname,':')) { /* Handle assign or volume name */
Al Viro29333922010-01-24 00:04:07 -050034 struct affs_sb_info *sbi = AFFS_SB(inode->i_sb);
35 char *pf;
36 spin_lock(&sbi->symlink_lock);
37 pf = sbi->s_prefix ? sbi->s_prefix : "/";
Linus Torvalds1da177e2005-04-16 15:20:36 -070038 while (i < 1023 && (c = pf[i]))
39 link[i++] = c;
Al Viro29333922010-01-24 00:04:07 -050040 spin_unlock(&sbi->symlink_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070041 while (i < 1023 && lf->symname[j] != ':')
42 link[i++] = lf->symname[j++];
43 if (i < 1023)
44 link[i++] = '/';
45 j++;
46 lc = '/';
47 }
48 while (i < 1023 && (c = lf->symname[j])) {
49 if (c == '/' && lc == '/' && i < 1020) { /* parent dir */
50 link[i++] = '.';
51 link[i++] = '.';
52 }
53 link[i++] = c;
54 lc = c;
55 j++;
56 }
57 link[i] = '\0';
58 affs_brelse(bh);
59 SetPageUptodate(page);
60 kunmap(page);
61 unlock_page(page);
62 return 0;
63fail:
64 SetPageError(page);
65 kunmap(page);
66 unlock_page(page);
Fabian Frederick196a4f82015-06-30 14:58:01 -070067 return -EIO;
Linus Torvalds1da177e2005-04-16 15:20:36 -070068}
69
Christoph Hellwigf5e54d62006-06-28 04:26:44 -070070const struct address_space_operations affs_symlink_aops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -070071 .readpage = affs_symlink_readpage,
72};
73
Arjan van de Ven754661f2007-02-12 00:55:38 -080074const struct inode_operations affs_symlink_inode_operations = {
Linus Torvalds1da177e2005-04-16 15:20:36 -070075 .readlink = generic_readlink,
76 .follow_link = page_follow_link_light,
77 .put_link = page_put_link,
78 .setattr = affs_notify_change,
79};