|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* | 
|  | *  linux/fs/affs/symlink.c | 
|  | * | 
|  | *  1995  Hans-Joachim Widmaier - Modified for affs. | 
|  | * | 
|  | *  Copyright (C) 1991, 1992  Linus Torvalds | 
|  | * | 
|  | *  affs symlink handling code | 
|  | */ | 
|  |  | 
|  | #include "affs.h" | 
|  |  | 
|  | static int affs_symlink_read_folio(struct file *file, struct folio *folio) | 
|  | { | 
|  | struct page *page = &folio->page; | 
|  | struct buffer_head *bh; | 
|  | struct inode *inode = page->mapping->host; | 
|  | char *link = page_address(page); | 
|  | struct slink_front *lf; | 
|  | int			 i, j; | 
|  | char			 c; | 
|  | char			 lc; | 
|  |  | 
|  | pr_debug("get_link(ino=%lu)\n", inode->i_ino); | 
|  |  | 
|  | bh = affs_bread(inode->i_sb, inode->i_ino); | 
|  | if (!bh) | 
|  | goto fail; | 
|  | i  = 0; | 
|  | j  = 0; | 
|  | lf = (struct slink_front *)bh->b_data; | 
|  | lc = 0; | 
|  |  | 
|  | if (strchr(lf->symname,':')) {	/* Handle assign or volume name */ | 
|  | struct affs_sb_info *sbi = AFFS_SB(inode->i_sb); | 
|  | char *pf; | 
|  | spin_lock(&sbi->symlink_lock); | 
|  | pf = sbi->s_prefix ? sbi->s_prefix : "/"; | 
|  | while (i < 1023 && (c = pf[i])) | 
|  | link[i++] = c; | 
|  | spin_unlock(&sbi->symlink_lock); | 
|  | while (i < 1023 && lf->symname[j] != ':') | 
|  | link[i++] = lf->symname[j++]; | 
|  | if (i < 1023) | 
|  | link[i++] = '/'; | 
|  | j++; | 
|  | lc = '/'; | 
|  | } | 
|  | while (i < 1023 && (c = lf->symname[j])) { | 
|  | if (c == '/' && lc == '/' && i < 1020) {	/* parent dir */ | 
|  | link[i++] = '.'; | 
|  | link[i++] = '.'; | 
|  | } | 
|  | link[i++] = c; | 
|  | lc = c; | 
|  | j++; | 
|  | } | 
|  | link[i] = '\0'; | 
|  | affs_brelse(bh); | 
|  | SetPageUptodate(page); | 
|  | unlock_page(page); | 
|  | return 0; | 
|  | fail: | 
|  | SetPageError(page); | 
|  | unlock_page(page); | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | const struct address_space_operations affs_symlink_aops = { | 
|  | .read_folio	= affs_symlink_read_folio, | 
|  | }; | 
|  |  | 
|  | const struct inode_operations affs_symlink_inode_operations = { | 
|  | .get_link	= page_get_link, | 
|  | .setattr	= affs_notify_change, | 
|  | }; |