Blame view

fs/hpfs/name.c 3.14 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
  /*
   *  linux/fs/hpfs/name.c
   *
   *  Mikulas Patocka (mikulas@artax.karlin.mff.cuni.cz), 1998-1999
   *
   *  operations with filenames
   */
  
  #include "hpfs_fn.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
  static inline int not_allowed_char(unsigned char c)
  {
  	return c<' ' || c=='"' || c=='*' || c=='/' || c==':' || c=='<' ||
  	      c=='>' || c=='?' || c=='\\' || c=='|';
  }
  
  static inline int no_dos_char(unsigned char c)
  {	/* Characters that are allowed in HPFS but not in DOS */
  	return c=='+' || c==',' || c==';' || c=='=' || c=='[' || c==']';
  }
  
  static inline unsigned char upcase(unsigned char *dir, unsigned char a)
  {
  	if (a<128 || a==255) return a>='a' && a<='z' ? a - 0x20 : a;
  	if (!dir) return a;
  	return dir[a-128];
  }
  
  unsigned char hpfs_upcase(unsigned char *dir, unsigned char a)
  {
  	return upcase(dir, a);
  }
  
  static inline unsigned char locase(unsigned char *dir, unsigned char a)
  {
  	if (a<128 || a==255) return a>='A' && a<='Z' ? a + 0x20 : a;
  	if (!dir) return a;
  	return dir[a];
  }
7e7742ee0   Al Viro   sanitize signedne...
39
  int hpfs_chk_name(const unsigned char *name, unsigned *len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
40
41
42
43
44
45
46
47
48
49
  {
  	int i;
  	if (*len > 254) return -ENAMETOOLONG;
  	hpfs_adjust_length(name, len);
  	if (!*len) return -EINVAL;
  	for (i = 0; i < *len; i++) if (not_allowed_char(name[i])) return -EINVAL;
  	if (*len == 1) if (name[0] == '.') return -EINVAL;
  	if (*len == 2) if (name[0] == '.' && name[1] == '.') return -EINVAL;
  	return 0;
  }
7e7742ee0   Al Viro   sanitize signedne...
50
  unsigned char *hpfs_translate_name(struct super_block *s, unsigned char *from,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
51
52
  			  unsigned len, int lc, int lng)
  {
7e7742ee0   Al Viro   sanitize signedne...
53
  	unsigned char *to;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
54
55
  	int i;
  	if (hpfs_sb(s)->sb_chk >= 2) if (hpfs_is_name_long(from, len) != lng) {
a19189e55   Fabian Frederick   fs/hpfs: increase...
56
  		pr_err("Long name flag mismatch - name ");
b7cb1ce22   Fabian Frederick   fs/hpfs: convert ...
57
58
59
60
  		for (i = 0; i < len; i++)
  			pr_cont("%c", from[i]);
  		pr_cont(" misidentified as %s.
  ", lng ? "short" : "long");
a19189e55   Fabian Frederick   fs/hpfs: increase...
61
62
63
  		pr_err("It's nothing serious. It could happen because of bug in OS/2.
  Set checks=normal to disable this message.
  ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
64
65
66
  	}
  	if (!lc) return from;
  	if (!(to = kmalloc(len, GFP_KERNEL))) {
a19189e55   Fabian Frederick   fs/hpfs: increase...
67
68
  		pr_err("can't allocate memory for name conversion buffer
  ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
69
70
71
72
73
  		return from;
  	}
  	for (i = 0; i < len; i++) to[i] = locase(hpfs_sb(s)->sb_cp_table,from[i]);
  	return to;
  }
7e7742ee0   Al Viro   sanitize signedne...
74
75
76
  int hpfs_compare_names(struct super_block *s,
  		       const unsigned char *n1, unsigned l1,
  		       const unsigned char *n2, unsigned l2, int last)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
77
78
79
80
81
82
83
84
85
86
87
88
89
90
  {
  	unsigned l = l1 < l2 ? l1 : l2;
  	unsigned i;
  	if (last) return -1;
  	for (i = 0; i < l; i++) {
  		unsigned char c1 = upcase(hpfs_sb(s)->sb_cp_table,n1[i]);
  		unsigned char c2 = upcase(hpfs_sb(s)->sb_cp_table,n2[i]);
  		if (c1 < c2) return -1;
  		if (c1 > c2) return 1;
  	}
  	if (l1 < l2) return -1;
  	if (l1 > l2) return 1;
  	return 0;
  }
7e7742ee0   Al Viro   sanitize signedne...
91
  int hpfs_is_name_long(const unsigned char *name, unsigned len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
92
93
94
95
96
97
98
99
100
101
102
103
  {
  	int i,j;
  	for (i = 0; i < len && name[i] != '.'; i++)
  		if (no_dos_char(name[i])) return 1;
  	if (!i || i > 8) return 1;
  	if (i == len) return 0;
  	for (j = i + 1; j < len; j++)
  		if (name[j] == '.' || no_dos_char(name[i])) return 1;
  	return j - i > 4;
  }
  
  /* OS/2 clears dots and spaces at the end of file name, so we have to */
7e7742ee0   Al Viro   sanitize signedne...
104
  void hpfs_adjust_length(const unsigned char *name, unsigned *len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
105
106
107
108
109
110
111
  {
  	if (!*len) return;
  	if (*len == 1 && name[0] == '.') return;
  	if (*len == 2 && name[0] == '.' && name[1] == '.') return;
  	while (*len && (name[*len - 1] == '.' || name[*len - 1] == ' '))
  		(*len)--;
  }