RISC-V: Fix a race condition during kernel stack overflow
[linux-block.git] / fs / affs / symlink.c
CommitLineData
b2441318 1// SPDX-License-Identifier: GPL-2.0
1da177e4
LT
2/*
3 * linux/fs/affs/symlink.c
4 *
5 * 1995 Hans-Joachim Widmaier - Modified for affs.
6 *
7 * Copyright (C) 1991, 1992 Linus Torvalds
8 *
9 * affs symlink handling code
10 */
11
12#include "affs.h"
13
1b6f3c87 14static int affs_symlink_read_folio(struct file *file, struct folio *folio)
1da177e4 15{
1b6f3c87 16 struct page *page = &folio->page;
1da177e4
LT
17 struct buffer_head *bh;
18 struct inode *inode = page->mapping->host;
21fc61c7 19 char *link = page_address(page);
1da177e4 20 struct slink_front *lf;
1da177e4
LT
21 int i, j;
22 char c;
23 char lc;
1da177e4 24
6b255391 25 pr_debug("get_link(ino=%lu)\n", inode->i_ino);
1da177e4 26
1da177e4
LT
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;
1da177e4
LT
34
35 if (strchr(lf->symname,':')) { /* Handle assign or volume name */
29333920
AV
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 : "/";
1da177e4
LT
40 while (i < 1023 && (c = pf[i]))
41 link[i++] = c;
29333920 42 spin_unlock(&sbi->symlink_lock);
1da177e4
LT
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);
1da177e4
LT
62 unlock_page(page);
63 return 0;
64fail:
65 SetPageError(page);
1da177e4 66 unlock_page(page);
196a4f82 67 return -EIO;
1da177e4
LT
68}
69
f5e54d6e 70const struct address_space_operations affs_symlink_aops = {
1b6f3c87 71 .read_folio = affs_symlink_read_folio,
1da177e4
LT
72};
73
754661f1 74const struct inode_operations affs_symlink_inode_operations = {
6b255391 75 .get_link = page_get_link,
1da177e4
LT
76 .setattr = affs_notify_change,
77};