diff options
| author | Eli Zaretskii | 2013-08-04 20:52:25 +0300 |
|---|---|---|
| committer | Eli Zaretskii | 2013-08-04 20:52:25 +0300 |
| commit | e443729d658ee2b9e0f55bbbb90241819bf516a6 (patch) | |
| tree | 99b2252965cb7586270cbe13d16726df82f25a8e /lib-src/ntlib.c | |
| parent | e0fdb6943066032db294720915c3bd644bf2bcd1 (diff) | |
| download | emacs-e443729d658ee2b9e0f55bbbb90241819bf516a6.tar.gz emacs-e443729d658ee2b9e0f55bbbb90241819bf516a6.zip | |
MS-Windows followup to last commit.
lib-src/ntlib.h: Include fcntl.h.
(mkostemp): Declare prototype.
(mktemp): Don't redefine.
lib-src/ntlib.c (mkostemp): New function.
Fixes: debbugs:15015
Diffstat (limited to 'lib-src/ntlib.c')
| -rw-r--r-- | lib-src/ntlib.c | 55 |
1 files changed, 55 insertions, 0 deletions
diff --git a/lib-src/ntlib.c b/lib-src/ntlib.c index 41b4e3a0cbc..0d0642d1bf2 100644 --- a/lib-src/ntlib.c +++ b/lib-src/ntlib.c | |||
| @@ -422,3 +422,58 @@ lstat (const char * path, struct stat * buf) | |||
| 422 | { | 422 | { |
| 423 | return stat (path, buf); | 423 | return stat (path, buf); |
| 424 | } | 424 | } |
| 425 | |||
| 426 | /* Implementation of mkostemp for MS-Windows, to avoid race conditions | ||
| 427 | when using mktemp. | ||
| 428 | |||
| 429 | Standard algorithm for generating a temporary file name seems to be | ||
| 430 | use pid or tid with a letter on the front (in place of the 6 X's) | ||
| 431 | and cycle through the letters to find a unique name. We extend | ||
| 432 | that to allow any reasonable character as the first of the 6 X's, | ||
| 433 | so that the number of simultaneously used temporary files will be | ||
| 434 | greater. */ | ||
| 435 | |||
| 436 | int | ||
| 437 | mkostemp (char * template, int flags) | ||
| 438 | { | ||
| 439 | char * p; | ||
| 440 | int i, fd = -1; | ||
| 441 | unsigned uid = GetCurrentThreadId (); | ||
| 442 | int save_errno = errno; | ||
| 443 | static char first_char[] = "abcdefghijklmnopqrstuvwyz0123456789!%-_@#"; | ||
| 444 | |||
| 445 | errno = EINVAL; | ||
| 446 | if (template == NULL) | ||
| 447 | return -1; | ||
| 448 | |||
| 449 | p = template + strlen (template); | ||
| 450 | i = 5; | ||
| 451 | /* replace up to the last 5 X's with uid in decimal */ | ||
| 452 | while (--p >= template && p[0] == 'X' && --i >= 0) | ||
| 453 | { | ||
| 454 | p[0] = '0' + uid % 10; | ||
| 455 | uid /= 10; | ||
| 456 | } | ||
| 457 | |||
| 458 | if (i < 0 && p[0] == 'X') | ||
| 459 | { | ||
| 460 | i = 0; | ||
| 461 | do | ||
| 462 | { | ||
| 463 | p[0] = first_char[i]; | ||
| 464 | if ((fd = open (template, | ||
| 465 | flags | _O_CREAT | _O_EXCL | _O_RDWR, | ||
| 466 | S_IRUSR | S_IWUSR)) >= 0 | ||
| 467 | || errno != EEXIST) | ||
| 468 | { | ||
| 469 | if (fd >= 0) | ||
| 470 | errno = save_errno; | ||
| 471 | return fd; | ||
| 472 | } | ||
| 473 | } | ||
| 474 | while (++i < sizeof (first_char)); | ||
| 475 | } | ||
| 476 | |||
| 477 | /* Template is badly formed or else we can't generate a unique name. */ | ||
| 478 | return -1; | ||
| 479 | } | ||