aboutsummaryrefslogtreecommitdiffstats
path: root/fs/udf
diff options
context:
space:
mode:
authorJan Kara <jack@suse.cz>2016-01-26 15:07:30 -0500
committerJan Kara <jack@suse.cz>2016-02-09 07:05:23 -0500
commit066b9cded00b8e3212df74a417bb074f3f3a1fe0 (patch)
treeff4eb1b93ce96a2b30956ee7b8c1b9569c3879c5 /fs/udf
parent9fba70569d9c3c253dba10ebbe3359f2157e504c (diff)
udf: Use separate buffer for copying split names
Code in udf_find_entry() and udf_readdir() used the same buffer for storing filename that was split among blocks and for the resulting filename in utf8. This worked because udf_get_filename() first internally copied the name into a different buffer and only then performed a conversion into the destination buffer. However we want to get rid of intermediate buffers so use separate buffer for converted name and name split between blocks so that we don't have the same source and destination buffer when converting split names. Signed-off-by: Jan Kara <jack@suse.cz>
Diffstat (limited to 'fs/udf')
-rw-r--r--fs/udf/dir.c13
-rw-r--r--fs/udf/namei.c13
2 files changed, 22 insertions, 4 deletions
diff --git a/fs/udf/dir.c b/fs/udf/dir.c
index 541d9c65014d..b51b371b874a 100644
--- a/fs/udf/dir.c
+++ b/fs/udf/dir.c
@@ -45,7 +45,7 @@ static int udf_readdir(struct file *file, struct dir_context *ctx)
45 int block, iblock; 45 int block, iblock;
46 loff_t nf_pos; 46 loff_t nf_pos;
47 int flen; 47 int flen;
48 unsigned char *fname = NULL; 48 unsigned char *fname = NULL, *copy_name = NULL;
49 unsigned char *nameptr; 49 unsigned char *nameptr;
50 uint16_t liu; 50 uint16_t liu;
51 uint8_t lfi; 51 uint8_t lfi;
@@ -143,7 +143,15 @@ static int udf_readdir(struct file *file, struct dir_context *ctx)
143 if (poffset >= lfi) { 143 if (poffset >= lfi) {
144 nameptr = (char *)(fibh.ebh->b_data + poffset - lfi); 144 nameptr = (char *)(fibh.ebh->b_data + poffset - lfi);
145 } else { 145 } else {
146 nameptr = fname; 146 if (!copy_name) {
147 copy_name = kmalloc(UDF_NAME_LEN,
148 GFP_NOFS);
149 if (!copy_name) {
150 ret = -ENOMEM;
151 goto out;
152 }
153 }
154 nameptr = copy_name;
147 memcpy(nameptr, fi->fileIdent + liu, 155 memcpy(nameptr, fi->fileIdent + liu,
148 lfi - poffset); 156 lfi - poffset);
149 memcpy(nameptr + lfi - poffset, 157 memcpy(nameptr + lfi - poffset,
@@ -185,6 +193,7 @@ out:
185 brelse(fibh.sbh); 193 brelse(fibh.sbh);
186 brelse(epos.bh); 194 brelse(epos.bh);
187 kfree(fname); 195 kfree(fname);
196 kfree(copy_name);
188 197
189 return ret; 198 return ret;
190} 199}
diff --git a/fs/udf/namei.c b/fs/udf/namei.c
index 9eb9c6440270..a2ba11eca995 100644
--- a/fs/udf/namei.c
+++ b/fs/udf/namei.c
@@ -165,7 +165,7 @@ static struct fileIdentDesc *udf_find_entry(struct inode *dir,
165 struct fileIdentDesc *fi = NULL; 165 struct fileIdentDesc *fi = NULL;
166 loff_t f_pos; 166 loff_t f_pos;
167 int block, flen; 167 int block, flen;
168 unsigned char *fname = NULL; 168 unsigned char *fname = NULL, *copy_name = NULL;
169 unsigned char *nameptr; 169 unsigned char *nameptr;
170 uint8_t lfi; 170 uint8_t lfi;
171 uint16_t liu; 171 uint16_t liu;
@@ -236,7 +236,15 @@ static struct fileIdentDesc *udf_find_entry(struct inode *dir,
236 nameptr = (uint8_t *)(fibh->ebh->b_data + 236 nameptr = (uint8_t *)(fibh->ebh->b_data +
237 poffset - lfi); 237 poffset - lfi);
238 else { 238 else {
239 nameptr = fname; 239 if (!copy_name) {
240 copy_name = kmalloc(UDF_NAME_LEN,
241 GFP_NOFS);
242 if (!copy_name) {
243 fi = ERR_PTR(-ENOMEM);
244 goto out_err;
245 }
246 }
247 nameptr = copy_name;
240 memcpy(nameptr, fi->fileIdent + liu, 248 memcpy(nameptr, fi->fileIdent + liu,
241 lfi - poffset); 249 lfi - poffset);
242 memcpy(nameptr + lfi - poffset, 250 memcpy(nameptr + lfi - poffset,
@@ -279,6 +287,7 @@ out_err:
279out_ok: 287out_ok:
280 brelse(epos.bh); 288 brelse(epos.bh);
281 kfree(fname); 289 kfree(fname);
290 kfree(copy_name);
282 291
283 return fi; 292 return fi;
284} 293}