ReactOS 0.4.16-dev-2473-gb34a1f1
lfn.h File Reference
This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Functions

void lfn_reset (void)
 
void lfn_add_slot (DIR_ENT *de, off_t dir_offset)
 
charlfn_get (DIR_ENT *de, off_t *lfn_offset)
 
void lfn_check_orphaned (void)
 
void lfn_fix_checksum (off_t from, off_t to, const char *short_name)
 

Function Documentation

◆ lfn_add_slot()

void lfn_add_slot ( DIR_ENT *  de,
off_t  dir_offset 
)

Definition at line 214 of file lfn.c.

215{
216 LFN_ENT *lfn = (LFN_ENT *) de;
217 int slot = lfn->id & LFN_ID_SLOTMASK;
218 unsigned offset;
219
220 if (lfn_slot == 0)
222
223 if (de->attr != VFAT_LN_ATTR)
224 die("lfn_add_slot called with non-LFN directory entry");
225
226 if (lfn->id & LFN_ID_START && slot != 0) {
227 if (lfn_slot != -1) {
228 int can_clear = 0;
229 /* There is already a LFN "in progess", so it is an error that a
230 * new start entry is here. */
231 /* Causes: 1) if slot# == expected: start bit set mysteriously, 2)
232 * old LFN overwritten by new one */
233 /* Fixes: 1) delete previous LFN 2) if slot# == expected and
234 * checksum ok: clear start bit */
235 /* XXX: Should delay that until next LFN known (then can better
236 * display the name) */
237 printf("A new long file name starts within an old one.\n");
238 if (slot == lfn_slot && lfn->alias_checksum == lfn_checksum) {
239 char *part1 = CNV_THIS_PART(lfn);
240 char *part2 = CNV_PARTS_SO_FAR();
241 printf(" It could be that the LFN start bit is wrong here\n"
242 " if \"%s\" seems to match \"%s\".\n", part1, part2);
243 free(part1);
244 free(part2);
245 can_clear = 1;
246 }
247 if (interactive) {
248 printf("1: Delete previous LFN\n2: Leave it as it is.\n");
249 if (can_clear)
250 printf("3: Clear start bit and concatenate LFNs\n");
251 } else
252 printf(" Not auto-correcting this.\n");
253 if (interactive) {
254 switch (get_key(can_clear ? "123" : "12", "?")) {
255 case '1':
257 lfn_reset();
258 break;
259 case '2':
260 break;
261 case '3':
262 lfn->id &= ~LFN_ID_START;
263 fs_write(dir_offset + offsetof(LFN_ENT, id),
264 sizeof(lfn->id), &lfn->id);
265 break;
266 }
267 }
268 }
269 lfn_slot = slot;
272 lfn_offsets = alloc(lfn_slot * sizeof(off_t));
273 lfn_parts = 0;
274 } else if (lfn_slot == -1 && slot != 0) {
275 /* No LFN in progress, but slot found; start bit missing */
276 /* Causes: 1) start bit got lost, 2) Previous slot with start bit got
277 * lost */
278 /* Fixes: 1) delete LFN, 2) set start bit */
279 char *part = CNV_THIS_PART(lfn);
280 printf("Long filename fragment \"%s\" found outside a LFN "
281 "sequence.\n (Maybe the start bit is missing on the "
282 "last fragment)\n", part);
283 free(part);
284 if (interactive) {
285 printf("1: Delete fragment\n2: Leave it as it is.\n"
286 "3: Set start bit\n");
287 } else
288 printf(" Not auto-correcting this.\n");
289 switch (interactive ? get_key("123", "?") : '2') {
290 case '1':
291 if (!lfn_offsets)
292 lfn_offsets = alloc(sizeof(off_t));
293 lfn_offsets[0] = dir_offset;
294 clear_lfn_slots(0, 0);
295 lfn_reset();
296 return;
297 case '2':
298 lfn_reset();
299 return;
300 case '3':
301 lfn->id |= LFN_ID_START;
302 fs_write(dir_offset + offsetof(LFN_ENT, id),
303 sizeof(lfn->id), &lfn->id);
304 lfn_slot = slot;
307 lfn_offsets = alloc(lfn_slot * sizeof(off_t));
308 lfn_parts = 0;
309 break;
310 }
311 } else if (slot != lfn_slot) {
312 /* wrong sequence number */
313 /* Causes: 1) seq-no destroyed */
314 /* Fixes: 1) delete LFN, 2) fix number (maybe only if following parts
315 * are ok?, maybe only if checksum is ok?) (Attention: space
316 * for name was allocated before!) */
317 int can_fix = 0;
318 printf("Unexpected long filename sequence number "
319 "(%d vs. expected %d).\n", slot, lfn_slot);
320 if (lfn->alias_checksum == lfn_checksum && lfn_slot > 0) {
321 char *part1 = CNV_THIS_PART(lfn);
322 char *part2 = CNV_PARTS_SO_FAR();
323 printf(" It could be that just the number is wrong\n"
324 " if \"%s\" seems to match \"%s\".\n", part1, part2);
325 free(part1);
326 free(part2);
327 can_fix = 1;
328 }
329 if (interactive) {
330 printf
331 ("1: Delete LFN\n2: Leave it as it is (and ignore LFN so far)\n");
332 if (can_fix)
333 printf("3: Correct sequence number\n");
334 } else
335 printf(" Not auto-correcting this.\n");
336 switch (interactive ? get_key(can_fix ? "123" : "12", "?") : '2') {
337 case '1':
338 if (!lfn_offsets) {
339 lfn_offsets = alloc(sizeof(off_t));
340 lfn_parts = 0;
341 }
342 lfn_offsets[lfn_parts++] = dir_offset;
344 lfn_reset();
345 return;
346 case '2':
347 lfn_reset();
348 return;
349 case '3':
350 lfn->id = (lfn->id & ~LFN_ID_SLOTMASK) | lfn_slot;
351 fs_write(dir_offset + offsetof(LFN_ENT, id),
352 sizeof(lfn->id), &lfn->id);
353 break;
354 }
355 }
356
357 if (lfn->alias_checksum != lfn_checksum) {
358 /* checksum mismatch */
359 /* Causes: 1) checksum field here destroyed */
360 /* Fixes: 1) delete LFN, 2) fix checksum */
361 printf("Checksum in long filename part wrong "
362 "(%02x vs. expected %02x).\n",
364 if (interactive) {
365 printf("1: Delete LFN\n2: Leave it as it is.\n"
366 "3: Correct checksum\n");
367 } else
368 printf(" Not auto-correcting this.\n");
369 if (interactive) {
370 switch (get_key("123", "?")) {
371 case '1':
372 lfn_offsets[lfn_parts++] = dir_offset;
374 lfn_reset();
375 return;
376 case '2':
377 break;
378 case '3':
380 fs_write(dir_offset + offsetof(LFN_ENT, alias_checksum),
381 sizeof(lfn->alias_checksum), &lfn->alias_checksum);
382 break;
383 }
384 }
385 }
386
387 if (lfn_slot != -1) {
388 lfn_slot--;
391 if (lfn->id & LFN_ID_START)
392 lfn_unicode[offset + 26] = lfn_unicode[offset + 27] = 0;
393 lfn_offsets[lfn_parts++] = dir_offset;
394 }
395
396 if (lfn->reserved != 0) {
397 printf("Reserved field in VFAT long filename slot is not 0 "
398 "(but 0x%02x).\n", lfn->reserved);
399 if (interactive)
400 printf("1: Fix.\n2: Leave it.\n");
401 else
402 printf("Auto-setting to 0.\n");
403 if (!interactive || get_key("12", "?") == '1') {
404 lfn->reserved = 0;
405 fs_write(dir_offset + offsetof(LFN_ENT, reserved),
406 sizeof(lfn->reserved), &lfn->reserved);
407 }
408 }
409 if (lfn->start != htole16(0)) {
410 printf("Start cluster field in VFAT long filename slot is not 0 "
411 "(but 0x%04x).\n", lfn->start);
412 if (interactive)
413 printf("1: Fix.\n2: Leave it.\n");
414 else
415 printf("Auto-setting to 0.\n");
416 if (!interactive || get_key("12", "?") == '1') {
417 lfn->start = htole16(0);
418 fs_write(dir_offset + offsetof(LFN_ENT, start),
419 sizeof(lfn->start), &lfn->start);
420 }
421 }
422}
#define free
Definition: debug_ros.c:5
r reserved
Definition: btrfs.c:3006
__kernel_off_t off_t
Definition: linux.h:201
#define printf
Definition: freeldr.h:104
#define VFAT_LN_ATTR
Definition: fsck.fat.h:78
GLuint start
Definition: gl.h:1545
GLintptr offset
Definition: glext.h:5920
#define CNV_PARTS_SO_FAR()
Definition: lfn.c:101
#define CNV_THIS_PART(lfn)
Definition: lfn.c:84
int lfn_slot
Definition: lfn.c:56
int lfn_parts
Definition: lfn.c:58
#define CHARS_PER_LFN
Definition: lfn.c:51
#define LFN_ID_SLOTMASK
Definition: lfn.c:49
off_t * lfn_offsets
Definition: lfn.c:57
unsigned char lfn_checksum
Definition: lfn.c:55
static void clear_lfn_slots(int start, int end)
Definition: lfn.c:172
void lfn_check_orphaned(void)
Definition: lfn.c:525
static void copy_lfn_part(unsigned char *dst, LFN_ENT *lfn)
Definition: lfn.c:165
void lfn_reset(void)
Definition: lfn.c:201
#define LFN_ID_START
Definition: lfn.c:48
unsigned char * lfn_unicode
Definition: lfn.c:54
#define die(str)
Definition: mkdosfs.c:347
#define alloc
Definition: rosglue.h:13
#define interactive
Definition: rosglue.h:34
#define offsetof(TYPE, MEMBER)
char get_key(const char *valid, const char *prompt)
Definition: common.c:184
void fs_write(off_t pos, int size, void *data)
Definition: io.c:344
#define htole16(x)
Definition: storage32.h:546
Definition: lfn.c:33
uint8_t id
Definition: lfn.c:34
uint8_t reserved
Definition: lfn.c:37
uint16_t start
Definition: lfn.c:40
uint8_t alias_checksum
Definition: lfn.c:38
Definition: vfat.h:185
struct _slot slot
Definition: vfat.h:196

Referenced by add_file().

◆ lfn_check_orphaned()

void lfn_check_orphaned ( void  )

Definition at line 525 of file lfn.c.

526{
527 char *long_name;
528
529 if (lfn_slot == -1)
530 return;
531
533 printf("Orphaned long file name part \"%s\"\n", long_name);
535 if (interactive)
536 printf("1: Delete.\n2: Leave it.\n");
537#ifdef __REACTOS__
538 else if (rw)
539#else
540 else
541#endif
542 printf(" Auto-deleting.\n");
543#ifdef __REACTOS__
544 if ((!interactive && rw) || (interactive && get_key("12", "?") == '1')) {
545#else
546 if (!interactive || get_key("12", "?") == '1') {
547#endif
549 }
550 lfn_reset();
551}
const WCHAR * long_name
Definition: reg.c:30
#define rw
Definition: rosglue.h:38

Referenced by add_file(), lfn_add_slot(), scan_dir(), and scan_root().

◆ lfn_fix_checksum()

void lfn_fix_checksum ( off_t  from,
off_t  to,
const char short_name 
)

Definition at line 189 of file lfn.c.

190{
191 int i;
192 uint8_t sum;
193 for (sum = 0, i = 0; i < 11; i++)
194 sum = (((sum & 1) << 7) | ((sum & 0xfe) >> 1)) + short_name[i];
195
196 for (; from < to; from += sizeof(LFN_ENT)) {
197 fs_write(from + offsetof(LFN_ENT, alias_checksum), sizeof(sum), &sum);
198 }
199}
const WCHAR * short_name
Definition: reg.c:29
unsigned char uint8_t
Definition: stdint.h:33
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint i
Definition: glfuncs.h:248
static int sum(int x_, int y_)
Definition: ptr2_test.cpp:35
CardRegion * from
Definition: spigame.cpp:19

Referenced by auto_rename(), and rename_file().

◆ lfn_get()

char * lfn_get ( DIR_ENT *  de,
off_t lfn_offset 
)

Definition at line 426 of file lfn.c.

427{
428 char *lfn;
429 uint8_t sum;
430 int i;
431
432 *lfn_offset = 0;
433 if (de->attr == VFAT_LN_ATTR)
434 die("lfn_get called with LFN directory entry");
435
436#if 0
437 if (de->lcase)
438 printf("lcase=%02x\n", de->lcase);
439#endif
440
441 if (lfn_slot == -1)
442 /* no long name for this file */
443 return NULL;
444
445 if (lfn_slot != 0) {
446 /* The long name isn't finished yet. */
447 /* Causes: 1) LFN slot overwritten by non-VFAT aware tool */
448 /* Fixes: 1) delete LFN 2) move overwriting entry to somewhere else
449 * and let user enter missing part of LFN (hard to do :-()
450 * 3) renumber entries and truncate name */
451 char *long_name = CNV_PARTS_SO_FAR();
452 char *short_name = file_name(de->name);
453 printf("Unfinished long file name \"%s\".\n"
454 " (Start may have been overwritten by %s)\n",
457 if (interactive) {
458 printf("1: Delete LFN\n2: Leave it as it is.\n"
459 "3: Fix numbering (truncates long name and attaches "
460 "it to short name %s)\n", short_name);
461 } else
462 printf(" Not auto-correcting this.\n");
463 switch (interactive ? get_key("123", "?") : '2') {
464 case '1':
466 lfn_reset();
467 return NULL;
468 case '2':
469 lfn_reset();
470 return NULL;
471 case '3':
472 for (i = 0; i < lfn_parts; ++i) {
473 uint8_t id = (lfn_parts - i) | (i == 0 ? LFN_ID_START : 0);
475 sizeof(id), &id);
476 }
479 break;
480 }
481 }
482
483 for (sum = 0, i = 0; i < MSDOS_NAME; i++)
484 sum = (((sum & 1) << 7) | ((sum & 0xfe) >> 1)) + de->name[i];
485 if (sum != lfn_checksum) {
486 /* checksum doesn't match, long name doesn't apply to this alias */
487 /* Causes: 1) alias renamed */
488 /* Fixes: 1) Fix checksum in LFN entries */
489 char *long_name = CNV_PARTS_SO_FAR();
490 char *short_name = file_name(de->name);
491 printf("Wrong checksum for long file name \"%s\".\n"
492 " (Short name %s may have changed without updating the long name)\n",
495 if (interactive) {
496 printf("1: Delete LFN\n2: Leave it as it is.\n"
497 "3: Fix checksum (attaches to short name %s)\n", short_name);
498 } else
499 printf(" Not auto-correcting this.\n");
500 if (interactive) {
501 switch (get_key("123", "?")) {
502 case '1':
504 lfn_reset();
505 return NULL;
506 case '2':
507 lfn_reset();
508 return NULL;
509 case '3':
510 for (i = 0; i < lfn_parts; ++i) {
511 fs_write(lfn_offsets[i] + offsetof(LFN_ENT, alias_checksum),
512 sizeof(sum), &sum);
513 }
514 break;
515 }
516 }
517 }
518
519 *lfn_offset = lfn_offsets[0];
521 lfn_reset();
522 return (lfn);
523}
#define NULL
Definition: types.h:112
#define UNTIL_0
Definition: lfn.c:75
static char * cnv_unicode(const unsigned char *uni, int maxlen, int use_q)
Definition: lfn.c:124
#define memmove(s1, s2, n)
Definition: mkisofs.h:881
static LPCWSTR file_name
Definition: protocol.c:147
#define MSDOS_NAME
Definition: msdos_fs.h:46

Referenced by add_file().

◆ lfn_reset()

void lfn_reset ( void  )

Definition at line 201 of file lfn.c.

202{
203 if (lfn_unicode)
206 if (lfn_offsets)
209 lfn_slot = -1;
210}

Referenced by lfn_add_slot(), lfn_check_orphaned(), lfn_get(), and new_dir().