VirtualBox

source: kBuild/trunk/src/kObjCache/kObjCache.c@ 1036

Last change on this file since 1036 was 1036, checked in by bird, 18 years ago

Ignore closing curly braces (extern "C" { ... }). Updated VCC statement.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 57.3 KB
Line 
1/* $Id: kObjCache.c 1036 2007-06-04 17:20:29Z bird $ */
2/** @file
3 *
4 * kObjCache - Object Cache.
5 *
6 * Copyright (c) 2007 knut st. osmundsen <[email protected]>
7 *
8 *
9 * This file is part of kBuild.
10 *
11 * kBuild is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; either version 2 of the License, or
14 * (at your option) any later version.
15 *
16 * kBuild is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License
22 * along with kBuild; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 *
25 */
26
27
28/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#include <string.h>
32#include <stdlib.h>
33#include <stdarg.h>
34#include <stdio.h>
35#include <errno.h>
36#include <assert.h>
37#include <sys/stat.h>
38#include <fcntl.h>
39#include <limits.h>
40#include <ctype.h>
41#ifndef PATH_MAX
42# define PATH_MAX _MAX_PATH /* windows */
43#endif
44#if defined(__OS2__) || defined(__WIN__)
45# include <process.h>
46# include <io.h>
47# ifdef __OS2__
48# include <unistd.h>
49# endif
50#else
51# include <unistd.h>
52# include <sys/wait.h>
53# ifndef O_BINARY
54# define O_BINARY 0
55# endif
56#endif
57#include "crc32.h"
58#include "md5.h"
59
60/*******************************************************************************
61* Defined Constants And Macros *
62*******************************************************************************/
63/** The max line length in a cache file. */
64#define KOBJCACHE_MAX_LINE_LEN 16384
65#if defined(__WIN__)
66# define PATH_SLASH '\\'
67#else
68# define PATH_SLASH '/'
69#endif
70#if defined(__OS2__) || defined(__WIN__)
71# define IS_SLASH(ch) ((ch) == '/' || (ch) == '\\')
72# define IS_SLASH_DRV(ch) ((ch) == '/' || (ch) == '\\' || (ch) == ':')
73#else
74# define IS_SLASH(ch) ((ch) == '/')
75# define IS_SLASH_DRV(ch) ((ch) == '/')
76#endif
77/** Use pipe instead of temp files when possible (speed). */
78#define USE_PIPE 1
79
80
81
82/*******************************************************************************
83* Structures and Typedefs *
84*******************************************************************************/
85/** A checksum list entry.
86 * We keep a list checksums (of precompiler output) that matches, The planned
87 * matching algorithm doesn't require the precompiler output to be indentical,
88 * only to produce the same object files.
89 */
90typedef struct KOCSUM
91{
92 /** The next checksum. */
93 struct KOCSUM *pNext;
94 /** The crc32 checksum. */
95 uint32_t crc32;
96 /** The MD5 digest. */
97 unsigned char md5[16];
98} KOCSUM, *PKOCSUM;
99/** Pointer to a const KOCSUM. */
100typedef const KOCSUM *PCKOCSUM;
101
102/**
103 * The object cache data.
104 */
105typedef struct KOBJCACHE
106{
107 /** The cache dir that all other names are relative to. */
108 char *pszDir;
109 /** The name of the cache file. */
110 const char *pszName;
111 /** Set if the object needs to be (re)compiled. */
112 unsigned fNeedCompiling;
113 /** Whether the precompiler runs in piped mode. If clear it's file
114 * mode (it could be redirected stdout, but that's essentially the
115 * same from our point of view). */
116 unsigned fPiped;
117
118 /** The name of new precompiled output. */
119 const char *pszNewCppName;
120 /** Pointer to the 'mapping' of the new precompiled output. */
121 char *pszNewCppMapping;
122 /** The size of the new precompiled output 'mapping'. */
123 size_t cbNewCpp;
124 /** The new checksum. */
125 KOCSUM NewSum;
126 /** The new object filename (relative to the cache file). */
127 char *pszNewObjName;
128
129 /** The name of the precompiled output. (relative to the cache file) */
130 char *pszOldCppName;
131 /** Pointer to the 'mapping' of the old precompiled output. */
132 char *pszOldCppMapping;
133 /** The size of the old precompiled output. */
134 size_t cbOldCpp;
135
136 /** The head of the checksum list. */
137 KOCSUM SumHead;
138 /** The object filename (relative to the cache file). */
139 char *pszObjName;
140 /** The compile argument vector used to build the object. */
141 char **papszArgvCompile;
142 /** The size of the compile */
143 unsigned cArgvCompile;
144} KOBJCACHE, *PKOBJCACHE;
145/** Pointer to a const KOBJCACHE. */
146typedef const KOBJCACHE *PCKOBJCACHE;
147
148
149/*******************************************************************************
150* Global Variables *
151*******************************************************************************/
152/** Whether verbose output is enabled. */
153static int g_fVerbose = 0;
154
155
156/*******************************************************************************
157* Internal Functions *
158*******************************************************************************/
159static const char *FindFilenameInPath(const char *pszPath);
160static char *AbsPath(const char *pszPath);
161static char *MakePathFromDirAndFile(const char *pszName, const char *pszDir);
162static char *CalcRelativeName(const char *pszPath, const char *pszDir);
163static FILE *FOpenFileInDir(const char *pszName, const char *pszDir, const char *pszMode);
164static int UnlinkFileInDir(const char *pszName, const char *pszDir);
165static int RenameFileInDir(const char *pszOldName, const char *pszNewName, const char *pszDir);
166static int DoesFileInDirExist(const char *pszName, const char *pszDir);
167static void *ReadFileInDir(const char *pszName, const char *pszDir, size_t *pcbFile);
168static void *xmalloc(size_t);
169static void *xrealloc(void *, size_t);
170static char *xstrdup(const char *);
171
172
173/**
174 * Compares two check sum entries.
175 *
176 * @returns 1 if equal, 0 if not equal.
177 *
178 * @param pSum1 The first checksum.
179 * @param pSum2 The second checksum.
180 */
181static int kObjCacheSumIsEqual(PCKOCSUM pSum1, PCKOCSUM pSum2)
182{
183 if (pSum1 == pSum2)
184 return 1;
185 if (!pSum1 || !pSum2)
186 return 0;
187 if (pSum1->crc32 != pSum2->crc32)
188 return 0;
189 if (memcmp(&pSum1->md5[0], &pSum2->md5[0], sizeof(pSum1->md5)))
190 return 0;
191 return 1;
192}
193
194
195/**
196 * Print a fatal error message and exit with rc=1.
197 *
198 * @param pEntry The cache entry.
199 * @param pszFormat The message to print.
200 * @param ... Format arguments.
201 */
202static void kObjCacheFatal(PCKOBJCACHE pEntry, const char *pszFormat, ...)
203{
204 va_list va;
205
206 fprintf(stderr, "kObjCache %s - fatal error: ", pEntry->pszName);
207 va_start(va, pszFormat);
208 vfprintf(stderr, pszFormat, va);
209 va_end(va);
210
211 exit(1);
212}
213
214
215/**
216 * Print a verbose message if verbosisty is enabled.
217 *
218 * @param pEntry The cache entry.
219 * @param pszFormat The message to print.
220 * @param ... Format arguments.
221 */
222static void kObjCacheVerbose(PCKOBJCACHE pEntry, const char *pszFormat, ...)
223{
224 if (g_fVerbose)
225 {
226 va_list va;
227
228 fprintf(stdout, "kObjCache %s - info: ", pEntry->pszName);
229 va_start(va, pszFormat);
230 vfprintf(stdout, pszFormat, va);
231 va_end(va);
232 }
233}
234
235
236/**
237 * Creates a cache entry for the given cache file name.
238 *
239 * @returns Pointer to a cache entry.
240 * @param pszFilename The cache file name.
241 */
242static PKOBJCACHE kObjCacheCreate(const char *pszFilename)
243{
244 PKOBJCACHE pEntry;
245
246 /*
247 * Allocate an empty entry.
248 */
249 pEntry = xmalloc(sizeof(*pEntry));
250 memset(pEntry, 0, sizeof(*pEntry));
251
252 /*
253 * Setup the directory and cache file name.
254 */
255 pEntry->pszDir = AbsPath(pszFilename);
256 pEntry->pszName = FindFilenameInPath(pEntry->pszDir);
257 if (pEntry->pszDir == pEntry->pszName)
258 kObjCacheFatal(pEntry, "Failed to find abs path for '%s'!\n", pszFilename);
259 ((char *)pEntry->pszName)[-1] = '\0';
260
261 return pEntry;
262}
263
264
265#if 0 /* don't bother. */
266/**
267 * Destroys the cache entry freeing up all it's resources.
268 *
269 * @param pEntry The entry to free.
270 */
271static void kObjCacheDestroy(PKOBJCACHE pEntry)
272{
273 free(pEntry->pszDir);
274 free(pEntry->pszName);
275 while (pEntry->SumHead.pNext)
276 {
277 void *pv = pEntry->SumHead.pNext;
278 pEntry->SumHead.pNext = pEntry->SumHead.pNext->pNext;
279 if (pv != &pEntry->NewSum)
280 free(pv);
281 }
282 free(pEntry);
283}
284#endif
285
286
287/**
288 * Reads and parses the cache file.
289 *
290 * @param pEntry The entry to read it into.
291 */
292static void kObjCacheRead(PKOBJCACHE pEntry)
293{
294 static char s_szLine[KOBJCACHE_MAX_LINE_LEN + 16];
295 FILE *pFile;
296 pFile = FOpenFileInDir(pEntry->pszName, pEntry->pszDir, "rb");
297 if (pFile)
298 {
299 kObjCacheVerbose(pEntry, "reading cache file...\n");
300
301 /*
302 * Check the magic.
303 */
304 if ( !fgets(s_szLine, sizeof(s_szLine), pFile)
305 || strcmp(s_szLine, "magic=kObjCache-1\n"))
306 {
307 kObjCacheVerbose(pEntry, "bad cache file (magic)\n");
308 pEntry->fNeedCompiling = 1;
309 }
310 else
311 {
312 /*
313 * Parse the rest of the file (relaxed order).
314 */
315 unsigned i;
316 int fBad = 0;
317 int fBadBeforeMissing;
318 int fFirstSum = 1;
319 while (fgets(s_szLine, sizeof(s_szLine), pFile))
320 {
321 /* Split the line and drop the trailing newline. */
322 char *pszNl = strchr(s_szLine, '\n');
323 char *pszVal = strchr(s_szLine, '=');
324 if ((fBad = pszVal == NULL))
325 break;
326 if (pszNl)
327 *pszNl = '\0';
328 *pszVal++ = '\0';
329
330 /* string case on variable name */
331 if (!strcmp(s_szLine, "obj"))
332 {
333 if ((fBad = pEntry->pszObjName != NULL))
334 break;
335 pEntry->pszObjName = xstrdup(pszVal);
336 }
337 else if (!strcmp(s_szLine, "cpp"))
338 {
339 if ((fBad = pEntry->pszOldCppName != NULL))
340 break;
341 pEntry->pszOldCppName = xstrdup(pszVal);
342 }
343 else if (!strcmp(s_szLine, "cpp-size"))
344 {
345 char *pszNext;
346 if ((fBad = pEntry->cbOldCpp != 0))
347 break;
348 pEntry->cbOldCpp = strtoul(pszVal, &pszNext, 0);
349 if ((fBad = pszNext && *pszNext))
350 break;
351 }
352 else if (!strcmp(s_szLine, "cc-argc"))
353 {
354 if ((fBad = pEntry->papszArgvCompile != NULL))
355 break;
356 pEntry->cArgvCompile = atoi(pszVal); /* if wrong, we'll fail below. */
357 pEntry->papszArgvCompile = xmalloc((pEntry->cArgvCompile + 1) * sizeof(pEntry->papszArgvCompile[0]));
358 memset(pEntry->papszArgvCompile, 0, (pEntry->cArgvCompile + 1) * sizeof(pEntry->papszArgvCompile[0]));
359 }
360 else if (!strncmp(s_szLine, "cc-argv-#", sizeof("cc-argv-#") - 1))
361 {
362 char *pszNext;
363 unsigned i = strtoul(&s_szLine[sizeof("cc-argv-#") - 1], &pszNext, 0);
364 if ((fBad = i >= pEntry->cArgvCompile || pEntry->papszArgvCompile[i] || (pszNext && *pszNext)))
365 break;
366 pEntry->papszArgvCompile[i] = xstrdup(pszVal);
367 }
368 else if (!strcmp(s_szLine, "sum"))
369 {
370 KOCSUM Sum;
371 unsigned i;
372 char *pszNext;
373 char *pszMD5 = strchr(pszVal, ':');
374 if ((fBad = pszMD5 == NULL))
375 break;
376 *pszMD5++ = '\0';
377
378 /* crc32 */
379 Sum.crc32 = (uint32_t)strtoul(pszVal, &pszNext, 16);
380 if ((fBad = (pszNext && *pszNext)))
381 break;
382
383 /* md5 */
384 for (i = 0; i < sizeof(Sum.md5) * 2; i++)
385 {
386 unsigned char ch = pszMD5[i];
387 int x;
388 if ((unsigned char)(ch - '0') <= 9)
389 x = ch - '0';
390 else if ((unsigned char)(ch - 'a') <= 5)
391 x = ch - 'a' + 10;
392 else if ((unsigned char)(ch - 'A') <= 5)
393 x = ch - 'A' + 10;
394 else
395 {
396 fBad = 1;
397 break;
398 }
399 if (!(i & 1))
400 Sum.md5[i >> 1] = x << 4;
401 else
402 Sum.md5[i >> 1] |= x;
403 }
404 if (fBad)
405 break;
406
407 if (fFirstSum)
408 {
409 pEntry->SumHead = Sum;
410 pEntry->SumHead.pNext = NULL;
411 fFirstSum = 0;
412 }
413 else
414 {
415 Sum.pNext = pEntry->SumHead.pNext;
416 pEntry->SumHead.pNext = xmalloc(sizeof(Sum));
417 *pEntry->SumHead.pNext = Sum;
418 }
419 }
420 else
421 {
422 fBad = 1;
423 break;
424 }
425 } /* parse loop */
426
427 /*
428 * Did we find everything?
429 */
430 fBadBeforeMissing = fBad;
431 if ( !fBad
432 && ( !pEntry->papszArgvCompile
433 || !pEntry->pszObjName
434 || !pEntry->pszOldCppName
435 || fFirstSum))
436 fBad = 1;
437 if (!fBad)
438 for (i = 0; i < pEntry->cArgvCompile; i++)
439 if ((fBad = !pEntry->papszArgvCompile[i]))
440 break;
441 if (fBad)
442 kObjCacheVerbose(pEntry, "bad cache file (%s)\n", fBadBeforeMissing ? s_szLine : "missing stuff");
443 else if (ferror(pFile))
444 kObjCacheVerbose(pEntry, "cache file read error\n");
445 pEntry->fNeedCompiling = fBad;
446 }
447 fclose(pFile);
448 }
449 else
450 {
451 kObjCacheVerbose(pEntry, "no cache file\n");
452 pEntry->fNeedCompiling = 1;
453 }
454}
455
456
457/**
458 * Writes the cache file.
459 *
460 * @param pEntry The entry to write.
461 */
462static void kObjCacheWrite(PKOBJCACHE pEntry)
463{
464 FILE *pFile;
465 PCKOCSUM pSum;
466 unsigned i;
467
468 kObjCacheVerbose(pEntry, "writing cache file...\n");
469 pFile = FOpenFileInDir(pEntry->pszName, pEntry->pszDir, "wb");
470 if (!pFile)
471 kObjCacheFatal(pEntry, "Failed to open '%s' in '%s': %s\n",
472 pEntry->pszName, pEntry->pszDir, strerror(errno));
473
474#define CHECK_LEN(expr) \
475 do { int cch = expr; if (cch >= KOBJCACHE_MAX_LINE_LEN) kObjCacheFatal(pEntry, "Line too long: %d (max %d)\nexpr: %s\n", cch, KOBJCACHE_MAX_LINE_LEN, #expr); } while (0)
476
477 fprintf(pFile, "magic=kObjCache-1\n");
478 CHECK_LEN(fprintf(pFile, "obj=%s\n", pEntry->pszNewObjName ? pEntry->pszNewObjName : pEntry->pszObjName));
479 CHECK_LEN(fprintf(pFile, "cpp=%s\n", pEntry->pszNewCppName ? pEntry->pszNewCppName : pEntry->pszOldCppName));
480 CHECK_LEN(fprintf(pFile, "cpp-size=%lu\n", pEntry->pszNewCppName ? pEntry->cbNewCpp : pEntry->cbOldCpp));
481 CHECK_LEN(fprintf(pFile, "cc-argc=%u\n", pEntry->cArgvCompile));
482 for (i = 0; i < pEntry->cArgvCompile; i++)
483 CHECK_LEN(fprintf(pFile, "cc-argv-#%u=%s\n", i, pEntry->papszArgvCompile[i]));
484 for (pSum = pEntry->fNeedCompiling ? &pEntry->NewSum : &pEntry->SumHead;
485 pSum;
486 pSum = pSum->pNext)
487 fprintf(pFile, "sum=%#x:%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x\n",
488 pSum->crc32,
489 pSum->md5[0], pSum->md5[1], pSum->md5[2], pSum->md5[3],
490 pSum->md5[4], pSum->md5[5], pSum->md5[6], pSum->md5[7],
491 pSum->md5[8], pSum->md5[9], pSum->md5[10], pSum->md5[11],
492 pSum->md5[12], pSum->md5[13], pSum->md5[14], pSum->md5[15]);
493
494 if ( fflush(pFile) < 0
495 || ferror(pFile))
496 {
497 int iErr = errno;
498 fclose(pFile);
499 UnlinkFileInDir(pEntry->pszName, pEntry->pszDir);
500 kObjCacheFatal(pEntry, "Stream error occured while writing '%s' in '%s': %d (?)\n",
501 pEntry->pszName, pEntry->pszDir, strerror(iErr));
502 }
503 fclose(pFile);
504}
505
506
507/**
508 * Spawns a child in a synchronous fashion.
509 * Terminating on failure.
510 *
511 * @param papszArgv Argument vector. The cArgv element is NULL.
512 * @param cArgv The number of arguments in the vector.
513 */
514static void kObjCacheSpawn(PCKOBJCACHE pEntry, const char **papszArgv, unsigned cArgv, const char *pszMsg, const char *pszStdOut)
515{
516#if defined(__OS2__) || defined(__WIN__)
517 intptr_t rc;
518 int fdStdOut = -1;
519 if (pszStdOut)
520 {
521 int fdReDir;
522 fdStdOut = dup(1); /* dup2(1,-1) doesn't work right on windows */
523 close(1);
524 fdReDir = open(pszStdOut, O_CREAT | O_TRUNC | O_WRONLY, 0777);
525 if (fdReDir < 0)
526 kObjCacheFatal(pEntry, "%s - failed to create stdout redirection file '%s': %s\n",
527 pszMsg, pszStdOut, strerror(errno));
528
529 if (fdReDir != 1)
530 {
531 if (dup2(fdReDir, 1) < 0)
532 kObjCacheFatal(pEntry, "%s - dup2 failed: %s\n", pszMsg, strerror(errno));
533 close(fdReDir);
534 }
535 }
536
537 errno = 0;
538 rc = _spawnvp(_P_WAIT, papszArgv[0], papszArgv);
539 if (rc < 0)
540 kObjCacheFatal(pEntry, "%s - _spawnvp failed (rc=0x%p): %s\n", pszMsg, rc, strerror(errno));
541 if (rc > 0)
542 kObjCacheFatal(pEntry, "%s - failed rc=%d\n", pszMsg, (int)rc);
543 if (fdStdOut)
544 {
545 close(1);
546 fdStdOut = dup2(fdStdOut, 1);
547 close(fdStdOut);
548 }
549
550#else
551 int iStatus;
552 pid_t pidWait;
553 pid_t pid = fork();
554 if (!pid)
555 {
556 if (pszStdOut)
557 {
558 int fdReDir;
559
560 close(1);
561 fdReDir = open(pszStdOut, O_CREAT | O_TRUNC | O_WRONLY, 0777);
562 if (fdReDir < 0)
563 kObjCacheFatal(pEntry, "%s - failed to create stdout redirection file '%s': %s\n",
564 pszMsg, pszStdOut, strerror(errno));
565 if (fdReDir != 1)
566 {
567 if (dup2(fdReDir, 1) < 0)
568 kObjCacheFatal(pEntry, "%s - dup2 failed: %s\n", pszMsg, strerror(errno));
569 close(fdReDir);
570 }
571 }
572
573 execvp(papszArgv[0], (char **)papszArgv);
574 kObjCacheFatal(pEntry, "%s - execvp failed: %s\n",
575 pszMsg, strerror(errno));
576 }
577 if (pid == -1)
578 kObjCacheFatal(pEntry, "%s - fork() failed: %s\n", pszMsg, strerror(errno));
579
580 pidWait = waitpid(pid, &iStatus, 0);
581 while (pidWait < 0 && errno == EINTR)
582 pidWait = waitpid(pid, &iStatus, 0);
583 if (pidWait != pid)
584 kObjCacheFatal(pEntry, "%s - waitpid failed rc=%d: %s\n",
585 pszMsg, pidWait, strerror(errno));
586 if (!WIFEXITED(iStatus))
587 kObjCacheFatal(pEntry, "%s - abended (iStatus=%#x)\n", pszMsg, iStatus);
588 if (WEXITSTATUS(iStatus))
589 kObjCacheFatal(pEntry, "%s - failed with rc %d\n", pszMsg, WEXITSTATUS(iStatus));
590#endif
591 (void)cArgv;
592}
593
594
595#ifdef USE_PIPE
596/**
597 * Spawns a child in a synchronous fashion.
598 * Terminating on failure.
599 *
600 * @param papszArgv Argument vector. The cArgv element is NULL.
601 * @param cArgv The number of arguments in the vector.
602 */
603static void kObjCacheSpawnPipe(PCKOBJCACHE pEntry, const char **papszArgv, unsigned cArgv, const char *pszMsg, char **ppszOutput, size_t *pcchOutput)
604{
605 int fds[2];
606 int iStatus;
607#if defined(__WIN__)
608 intptr_t pid, pidWait;
609#else
610 pid_t pid, pidWait;
611#endif
612 int fdStdOut;
613 size_t cbAlloc;
614 size_t cbLeft;
615 char *psz;
616
617 /*
618 * Setup the pipe.
619 */
620#if defined(__WIN__)
621 if (_pipe(fds, 0, _O_NOINHERIT | _O_BINARY) < 0)
622#else
623 if (pipe(fds) < 0)
624#endif
625 kObjCacheFatal(pEntry, "pipe failed: %s\n", strerror(errno));
626 fdStdOut = dup(1);
627 if (dup2(fds[1 /* write */], 1) < 0)
628 kObjCacheFatal(pEntry, "dup2(,1) failed: %s\n", strerror(errno));
629 close(fds[1]);
630 fds[1] = -1;
631#ifndef __WIN__
632 fcntl(fds[0], F_SETFD, FD_CLOEXEC);
633 fcntl(fdStdOut, F_SETFD, FD_CLOEXEC);
634#endif
635
636 /*
637 * Create the child process.
638 */
639#if defined(__OS2__) || defined(__WIN__)
640 errno = 0;
641 pid = _spawnvp(_P_NOWAIT, papszArgv[0], papszArgv);
642 if (pid == -1)
643 kObjCacheFatal(pEntry, "%s - _spawnvp failed: %s\n", pszMsg, strerror(errno));
644
645#else
646 pid = fork();
647 if (!pid)
648 {
649 execvp(papszArgv[0], (char **)papszArgv);
650 kObjCacheFatal(pEntry, "%s - execvp failed: %s\n",
651 pszMsg, strerror(errno));
652 }
653 if (pid == -1)
654 kObjCacheFatal(pEntry, "%s - fork() failed: %s\n", pszMsg, strerror(errno));
655#endif
656
657 /*
658 * Restore stdout.
659 */
660 close(1);
661 fdStdOut = dup2(fdStdOut, 1);
662
663 /*
664 * Read data from the child.
665 */
666 cbAlloc = pEntry->cbOldCpp ? (pEntry->cbOldCpp + 4*1024*1024 + 4096) & ~(4*1024*1024 - 1) : 4*1024*1024;
667 cbLeft = cbAlloc;
668 *ppszOutput = psz = xmalloc(cbAlloc);
669 for (;;)
670 {
671 long cbRead = read(fds[0], psz, cbLeft - 1);
672 if (!cbRead)
673 break;
674 if (cbRead < 0 && errno != EINTR)
675 kObjCacheFatal(pEntry, "%s - read(%d,,%ld) failed: %s\n", pszMsg, fds[0], (long)cbLeft, strerror(errno));
676 psz += cbRead;
677 *psz = '\0';
678 cbLeft -= cbRead;
679
680 /* expand the buffer? */
681 if (cbLeft <= 1)
682 {
683 size_t off = psz - *ppszOutput;
684 assert(off == cbAlloc);
685 cbLeft = 4*1024*1024;
686 cbAlloc += cbLeft;
687 *ppszOutput = xrealloc(*ppszOutput, cbAlloc);
688 psz = *ppszOutput + off;
689 }
690 }
691 close(fds[0]);
692 *pcchOutput = cbAlloc - cbLeft;
693
694 /*
695 * Reap the child.
696 */
697#ifdef __WIN__
698 pidWait = _cwait(&iStatus, pid, _WAIT_CHILD);
699 if (pidWait == -1)
700 kObjCacheFatal(pEntry, "%s - waitpid failed: %s\n",
701 pszMsg, strerror(errno));
702 if (iStatus)
703 kObjCacheFatal(pEntry, "%s - failed with rc %d\n", pszMsg, iStatus);
704#else
705 pidWait = waitpid(pid, &iStatus, 0);
706 while (pidWait < 0 && errno == EINTR)
707 pidWait = waitpid(pid, &iStatus, 0);
708 if (pidWait != pid)
709 kObjCacheFatal(pEntry, "%s - waitpid failed rc=%d: %s\n",
710 pszMsg, pidWait, strerror(errno));
711 if (!WIFEXITED(iStatus))
712 kObjCacheFatal(pEntry, "%s - abended (iStatus=%#x)\n", pszMsg, iStatus);
713 if (WEXITSTATUS(iStatus))
714 kObjCacheFatal(pEntry, "%s - failed with rc %d\n", pszMsg, WEXITSTATUS(iStatus));
715#endif
716 (void)cArgv;
717}
718#endif /* USE_PIPE */
719
720
721/**
722 * Reads the (new) output of the precompiler.
723 *
724 * Not used when using pipes.
725 *
726 * @param pEntry The cache entry. cbNewCpp and pszNewCppMapping will be updated.
727 */
728static void kObjCacheReadPrecompileOutput(PKOBJCACHE pEntry)
729{
730 pEntry->pszNewCppMapping = ReadFileInDir(pEntry->pszNewCppName, pEntry->pszDir, &pEntry->cbNewCpp);
731 if (!pEntry->pszNewCppMapping)
732 kObjCacheFatal(pEntry, "failed to open/read '%s' in '%s': %s\n",
733 pEntry->pszNewCppName, pEntry->pszDir, strerror(errno));
734 kObjCacheVerbose(pEntry, "precompiled file is %lu bytes long\n", (unsigned long)pEntry->cbNewCpp);
735}
736
737
738/**
739 * Worker for kObjCachePreCompile and calculates the checksum of
740 * the precompiler output.
741 *
742 * @param pEntry The cache entry. NewSum will be updated.
743 */
744static void kObjCacheCalcChecksum(PKOBJCACHE pEntry)
745{
746 struct MD5Context MD5Ctx;
747
748 memset(&pEntry->NewSum, 0, sizeof(pEntry->NewSum));
749 pEntry->NewSum.crc32 = crc32(0, pEntry->pszNewCppMapping, pEntry->cbNewCpp);
750 MD5Init(&MD5Ctx);
751 MD5Update(&MD5Ctx, (unsigned char *)pEntry->pszNewCppMapping, pEntry->cbNewCpp);
752 MD5Final(&pEntry->NewSum.md5[0], &MD5Ctx);
753 kObjCacheVerbose(pEntry, "crc32=%#lx md5=%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x\n",
754 pEntry->NewSum.crc32,
755 pEntry->NewSum.md5[0], pEntry->NewSum.md5[1], pEntry->NewSum.md5[2], pEntry->NewSum.md5[3],
756 pEntry->NewSum.md5[4], pEntry->NewSum.md5[5], pEntry->NewSum.md5[6], pEntry->NewSum.md5[7],
757 pEntry->NewSum.md5[8], pEntry->NewSum.md5[9], pEntry->NewSum.md5[10], pEntry->NewSum.md5[11],
758 pEntry->NewSum.md5[12], pEntry->NewSum.md5[13], pEntry->NewSum.md5[14], pEntry->NewSum.md5[15]);
759
760}
761
762
763/**
764 * Run the precompiler and calculate the checksum of the output.
765 *
766 * @param pEntry The cache entry.
767 * @param papszArgvPreComp The argument vector for executing precompiler. The cArgvPreComp'th argument must be NULL.
768 * @param cArgvPreComp The number of arguments.
769 * @param pszPreCompName Precompile output name. (must kick around)
770 * @param fRedirStdOut Whether stdout needs to be redirected or not.
771 */
772static void kObjCachePreCompile(PKOBJCACHE pEntry, const char **papszArgvPreComp, unsigned cArgvPreComp, const char *pszPreCompName, int fRedirStdOut)
773{
774#ifdef USE_PIPE
775 /*
776 * Flag it as piped or non-piped.
777 */
778 if (fRedirStdOut)
779 pEntry->fPiped = 1;
780 else
781#endif
782 pEntry->fPiped = 0;
783
784 /*
785 * Rename the old precompiled output to '-old'.
786 * We'll discard the old output and keep the new output, but because
787 * we might with to do a quick matchup later we can't remove it just now.
788 * If we're using the pipe strategy, we will not do any renaming.
789 */
790 if ( pEntry->pszOldCppName
791 && !pEntry->fPiped
792 && DoesFileInDirExist(pEntry->pszOldCppName, pEntry->pszDir))
793 {
794 size_t cch = strlen(pEntry->pszOldCppName);
795 char *psz = xmalloc(cch + sizeof("-old"));
796 memcpy(psz, pEntry->pszOldCppName, cch);
797 memcpy(psz + cch, "-old", sizeof("-old"));
798
799 kObjCacheVerbose(pEntry, "renaming '%s' to '%s' in '%s'\n", pEntry->pszOldCppName, psz, pEntry->pszDir);
800 UnlinkFileInDir(psz, pEntry->pszDir);
801 if (RenameFileInDir(pEntry->pszOldCppName, psz, pEntry->pszDir))
802 kObjCacheFatal(pEntry, "failed to rename '%s' -> '%s' in '%s': %s\n",
803 pEntry->pszOldCppName, psz, pEntry->pszDir, strerror(errno));
804 free(pEntry->pszOldCppName);
805 pEntry->pszOldCppName = psz;
806 }
807 pEntry->pszNewCppName = CalcRelativeName(pszPreCompName, pEntry->pszDir);
808
809 /*
810 * Precompile it and calculate the checksum on the output.
811 */
812 kObjCacheVerbose(pEntry, "precompiling -> '%s'...\n", pEntry->pszNewCppName);
813#ifdef USE_PIPE
814 if (pEntry->fPiped)
815 kObjCacheSpawnPipe(pEntry, papszArgvPreComp, cArgvPreComp, "precompile", &pEntry->pszNewCppMapping, &pEntry->cbNewCpp);
816 else
817#endif
818 {
819 if (fRedirStdOut)
820 kObjCacheSpawn(pEntry, papszArgvPreComp, cArgvPreComp, "precompile", pszPreCompName);
821 else
822 kObjCacheSpawn(pEntry, papszArgvPreComp, cArgvPreComp, "precompile", NULL);
823 kObjCacheReadPrecompileOutput(pEntry);
824 }
825 kObjCacheCalcChecksum(pEntry);
826}
827
828
829/**
830 * Check whether the string is a '#line' statement.
831 *
832 * @returns 1 if it is, 0 if it isn't.
833 * @param psz The line to examin.
834 * @parma piLine Where to store the line number.
835 * @parma ppszFile Where to store the start of the filename.
836 */
837static int IsLineStatement(const char *psz, unsigned *piLine, const char **ppszFile)
838{
839 unsigned iLine;
840
841 /* Expect a hash. */
842 if (*psz++ != '#')
843 return 0;
844
845 /* Skip blanks between '#' and the line / number */
846 while (*psz == ' ' || *psz == '\t')
847 psz++;
848
849 /* Skip the 'line' if present. */
850 if (!strncmp(psz, "line", sizeof("line") - 1))
851 psz += sizeof("line");
852
853 /* Expect a line number now. */
854 if ((unsigned char)(*psz - '0') > 9)
855 return 0;
856 iLine = 0;
857 do
858 {
859 iLine *= 10;
860 iLine += (*psz - '0');
861 psz++;
862 }
863 while ((unsigned char)(*psz - '0') <= 9);
864
865 /* Expect one or more space now. */
866 if (*psz != ' ' && *psz != '\t')
867 return 0;
868 do psz++;
869 while (*psz == ' ' || *psz == '\t');
870
871 /* that's good enough. */
872 *piLine = iLine;
873 *ppszFile = psz;
874 return 1;
875}
876
877
878/**
879 * Scan backwards for the previous #line statement.
880 *
881 * @returns The filename in the previous statement.
882 * @param pszStart Where to start.
883 * @param pszStop Where to stop. Less than pszStart.
884 * @param piLine The line number count to adjust.
885 */
886static const char *FindFileStatement(const char *pszStart, const char *pszStop, unsigned *piLine)
887{
888 unsigned iLine = *piLine;
889 assert(pszStart >= pszStop);
890 while (pszStart >= pszStop)
891 {
892 if (*pszStart == '\n')
893 iLine++;
894 else if (*pszStart == '#')
895 {
896 unsigned iLineTmp;
897 const char *pszFile;
898 const char *psz = pszStart - 1;
899 while (psz >= pszStop && (*psz == ' ' || *psz =='\t'))
900 psz--;
901 if ( (psz < pszStop || *psz == '\n')
902 && IsLineStatement(pszStart, &iLineTmp, &pszFile))
903 {
904 *piLine = iLine + iLineTmp - 1;
905 return pszFile;
906 }
907 }
908 pszStart--;
909 }
910 return NULL;
911}
912
913
914/**
915 * Worker for kObjCacheCompareOldAndNewOutput() that compares the
916 * precompiled output using a fast but not very good method.
917 *
918 * @returns 1 if matching, 0 if not matching.
919 * @param pEntry The entry containing the names of the files to compare.
920 * The entry is not updated in any way.
921 */
922static int kObjCacheCompareFast(PCKOBJCACHE pEntry)
923{
924 const char * psz1 = pEntry->pszNewCppMapping;
925 const char * const pszEnd1 = psz1 + pEntry->cbNewCpp;
926 const char * psz2 = pEntry->pszOldCppMapping;
927 const char * const pszEnd2 = psz2 + pEntry->cbOldCpp;
928
929 assert(*pszEnd1 == '\0');
930 assert(*pszEnd2 == '\0');
931
932 /*
933 * Iterate block by block and backtrack when we find a difference.
934 */
935 for (;;)
936 {
937 size_t cch = pszEnd1 - psz1;
938 if (cch > (size_t)(pszEnd2 - psz2))
939 cch = pszEnd2 - psz2;
940 if (cch > 4096)
941 cch = 4096;
942 if ( cch
943 && !memcmp(psz1, psz2, cch))
944 {
945 /* no differences */
946 psz1 += cch;
947 psz2 += cch;
948 }
949 else
950 {
951 /*
952 * Pinpoint the difference exactly and the try find the start
953 * of that line. Then skip forward until we find something to
954 * work on that isn't spaces, #line statements or closing curly
955 * braces.
956 *
957 * The closing curly braces are ignored because they are frequently
958 * found at the end of header files (__END_DECLS) and the worst
959 * thing that may happen if it isn't one of these braces we're
960 * ignoring is that the final line in a function block is a little
961 * bit off in the debug info.
962 *
963 * Since we might be skipping a few new empty headers, it is
964 * possible that we will omit this header from the dependencies
965 * when using VCC. This might not be a problem, since it seems
966 * we'll have to use the precompiler output to generate the deps
967 * anyway.
968 */
969 const char *psz;
970 const char *pszMismatch1;
971 const char *pszFile1 = NULL;
972 unsigned iLine1 = 0;
973 unsigned cCurlyBraces1 = 0;
974 const char *pszMismatch2;
975 const char *pszFile2 = NULL;
976 unsigned iLine2 = 0;
977 unsigned cCurlyBraces2 = 0;
978
979 /* locate the difference. */
980 while (cch >= 512 && !memcmp(psz1, psz2, 512))
981 psz1 += 512, psz2 += 512, cch -= 512;
982 while (cch >= 64 && !memcmp(psz1, psz2, 64))
983 psz1 += 64, psz2 += 64, cch -= 64;
984 while (*psz1 == *psz2 && cch > 0)
985 psz1++, psz2++, cch--;
986
987 /* locate the start of that line. */
988 psz = psz1;
989 while ( psz > pEntry->pszNewCppMapping
990 && psz[-1] != '\n')
991 psz--;
992 psz2 -= (psz1 - psz);
993 pszMismatch2 = psz2;
994 pszMismatch1 = psz1 = psz;
995
996 /* Parse the 1st file line by line. */
997 while (psz1 < pszEnd1)
998 {
999 if (*psz1 == '\n')
1000 {
1001 psz1++;
1002 iLine1++;
1003 }
1004 else
1005 {
1006 psz = psz1;
1007 while (isspace(*psz) && *psz != '\n')
1008 psz++;
1009 if (*psz == '\n')
1010 {
1011 psz1 = psz + 1;
1012 iLine1++;
1013 }
1014 else if (*psz == '#' && IsLineStatement(psz, &iLine1, &pszFile1))
1015 {
1016 psz1 = memchr(psz, '\n', pszEnd1 - psz);
1017 if (!psz1++)
1018 psz1 = pszEnd1;
1019 }
1020 else if (*psz == '}')
1021 {
1022 do psz++;
1023 while (isspace(*psz) && *psz != '\n');
1024 if (*psz == '\n')
1025 iLine1++;
1026 else if (psz != pszEnd1)
1027 break;
1028 cCurlyBraces1++;
1029 psz1 = psz;
1030 }
1031 else if (psz == pszEnd1)
1032 psz1 = psz;
1033 else /* found something that can be compared. */
1034 break;
1035 }
1036 }
1037
1038 /* Ditto for the 2nd file. */
1039 while (psz2 < pszEnd2)
1040 {
1041 if (*psz2 == '\n')
1042 {
1043 psz2++;
1044 iLine2++;
1045 }
1046 else
1047 {
1048 psz = psz2;
1049 while (isspace(*psz) && *psz != '\n')
1050 psz++;
1051 if (*psz == '\n')
1052 {
1053 psz2 = psz + 1;
1054 iLine2++;
1055 }
1056 else if (*psz == '#' && IsLineStatement(psz, &iLine2, &pszFile2))
1057 {
1058 psz2 = memchr(psz, '\n', pszEnd2 - psz);
1059 if (!psz2++)
1060 psz2 = pszEnd2;
1061 }
1062 else if (*psz == '}')
1063 {
1064 do psz++;
1065 while (isspace(*psz) && *psz != '\n');
1066 if (*psz == '\n')
1067 iLine2++;
1068 else if (psz != pszEnd2)
1069 break;
1070 cCurlyBraces2++;
1071 psz2 = psz;
1072 }
1073 else if (psz == pszEnd2)
1074 psz2 = psz;
1075 else /* found something that can be compared. */
1076 break;
1077 }
1078 }
1079
1080 /* Match the number of ignored closing curly braces. */
1081 if (cCurlyBraces1 != cCurlyBraces2)
1082 return 0;
1083
1084 /* Reaching the end of any of them means the return statement can decide. */
1085 if ( psz1 == pszEnd1
1086 || psz2 == pszEnd2)
1087 break;
1088
1089 /* Match the current line. */
1090 psz = memchr(psz1, '\n', pszEnd1 - psz1);
1091 if (!psz)
1092 psz = pszEnd1;
1093 cch = psz - psz1;
1094 if (psz2 + cch > pszEnd2)
1095 break;
1096 if (memcmp(psz1, psz2, cch))
1097 break;
1098
1099 /* Check that we're at the same location now. */
1100 if (!pszFile1)
1101 pszFile1 = FindFileStatement(pszMismatch1, pEntry->pszNewCppMapping, &iLine1);
1102 if (!pszFile2)
1103 pszFile2 = FindFileStatement(pszMismatch2, pEntry->pszOldCppMapping, &iLine2);
1104 if (pszFile1 && pszFile2)
1105 {
1106 if (iLine1 != iLine2)
1107 break;
1108 while (*pszFile1 == *pszFile2 && *pszFile1 != '\n' && *pszFile1)
1109 pszFile1++, pszFile2++;
1110 if (*pszFile1 != *pszFile2)
1111 break;
1112 }
1113 else if (pszFile1 || pszFile2)
1114 {
1115 assert(0); /* this shouldn't happen. */
1116 break;
1117 }
1118
1119 /* Try align psz1 on 8 or 4 bytes so at least one of the buffers are aligned. */
1120 psz1 += cch;
1121 psz2 += cch;
1122 if (cch >= ((uintptr_t)psz1 & 7))
1123 {
1124 psz2 -= ((uintptr_t)psz1 & 7);
1125 psz1 -= ((uintptr_t)psz1 & 7);
1126 }
1127 else if (cch >= ((uintptr_t)psz1 & 3))
1128 {
1129 psz2 -= ((uintptr_t)psz1 & 3);
1130 psz1 -= ((uintptr_t)psz1 & 3);
1131 }
1132 }
1133 }
1134
1135 return psz1 == pszEnd1
1136 && psz2 == pszEnd2;
1137}
1138
1139
1140/**
1141 * Worker for kObjCacheCompileIfNeeded that compares the
1142 * precompiled output.
1143 *
1144 * @returns 1 if matching, 0 if not matching.
1145 * @param pEntry The entry containing the names of the files to compare.
1146 * This will load the old cpp output (changing pszOldCppName and cbOldCpp).
1147 */
1148static int kObjCacheCompareOldAndNewOutput(PKOBJCACHE pEntry)
1149{
1150 /** @todo do some quick but fancy comparing that determins whether code
1151 * has actually changed or moved. We can ignore declarations and typedefs that
1152 * has just been moved up/down a bit. The typical case is adding a new error
1153 * #define that doesn't influence the current compile job. */
1154
1155 /*
1156 * Load the old output.
1157 */
1158 pEntry->pszOldCppMapping = ReadFileInDir(pEntry->pszOldCppName, pEntry->pszDir, &pEntry->cbOldCpp);
1159 if (!pEntry->pszOldCppMapping)
1160 {
1161 kObjCacheVerbose(pEntry, "failed to read old cpp file ('%s' in '%s'): %s\n",
1162 pEntry->pszOldCppName, pEntry->pszDir, strerror(errno));
1163 return 0;
1164 }
1165
1166 /*
1167 * I may implement a more sophisticated alternative method later... maybe.
1168 */
1169 //if ()
1170 // return kObjCacheCompareBest(pEntry);
1171 return kObjCacheCompareFast(pEntry);
1172}
1173
1174
1175/**
1176 * Worker for kObjCacheCompileIfNeeded that does the actual (re)compilation.
1177 *
1178 * @returns 1 if matching, 0 if not matching.
1179 * @param pEntry The cache entry.
1180 * @param papszArgvCompile The argument vector for invoking the compiler. The cArgvCompile'th entry must be NULL.
1181 * @param cArgvCompile The number of arguments in the vector.
1182 * @param pszObjName The name of the object file.
1183 */
1184static void kObjCacheCompileIt(PKOBJCACHE pEntry, const char **papszArgvCompile, unsigned cArgvCompile, const char *pszObjName)
1185{
1186 /*
1187 * Delete the old object file and precompiler output.
1188 */
1189 if (pEntry->pszObjName)
1190 {
1191 UnlinkFileInDir(pEntry->pszObjName, pEntry->pszDir);
1192 pEntry->pszObjName = NULL;
1193 }
1194 pEntry->pszNewObjName = CalcRelativeName(pszObjName, pEntry->pszDir);
1195
1196 /*
1197 * If we executed the precompiled in piped mode we'll have to write the
1198 * precompiler output to disk so the compile has some thing to chew on.
1199 */
1200 if (pEntry->fPiped)
1201 {
1202 FILE *pFile = FOpenFileInDir(pEntry->pszNewCppName, pEntry->pszDir, "wb");
1203 if (!pFile)
1204 kObjCacheFatal(pEntry, "failed to create file '%s' in '%s': %s\n",
1205 pEntry->pszNewCppName, pEntry->pszDir, strerror(errno));
1206 if (fwrite(pEntry->pszNewCppMapping, pEntry->cbNewCpp, 1, pFile) != 1)
1207 kObjCacheFatal(pEntry, "fwrite failed: %s\n", strerror(errno));
1208 if (fclose(pFile))
1209 kObjCacheFatal(pEntry, "fclose failed: %s\n", strerror(errno));
1210 }
1211
1212 /*
1213 * Release buffers we no longer need before starting the compile.
1214 */
1215 free(pEntry->pszNewCppMapping);
1216 pEntry->pszNewCppMapping = NULL;
1217 free(pEntry->pszOldCppMapping);
1218 pEntry->pszOldCppMapping = NULL;
1219
1220 /*
1221 * Do the recompilation.
1222 */
1223 kObjCacheVerbose(pEntry, "compiling -> '%s'...\n", pEntry->pszNewObjName);
1224 pEntry->papszArgvCompile = (char **)papszArgvCompile; /* LEAK */
1225 pEntry->cArgvCompile = cArgvCompile;
1226 kObjCacheSpawn(pEntry, papszArgvCompile, cArgvCompile, "compile", NULL);
1227}
1228
1229
1230/**
1231 * Check if (re-)compilation is required and do it.
1232 *
1233 * @returns 1 if matching, 0 if not matching.
1234 * @param pEntry The cache entry.
1235 * @param papszArgvCompile The argument vector for invoking the compiler. The cArgvCompile'th entry must be NULL.
1236 * @param cArgvCompile The number of arguments in the vector.
1237 * @param pszObjName The name of the object file.
1238 */
1239static void kObjCacheCompileIfNeeded(PKOBJCACHE pEntry, const char **papszArgvCompile, unsigned cArgvCompile, const char *pszObjName)
1240{
1241 /*
1242 * Does the object name differ?
1243 */
1244 if (!pEntry->fNeedCompiling)
1245 {
1246 char *pszTmp = CalcRelativeName(pszObjName, pEntry->pszDir);
1247 if (strcmp(pEntry->pszObjName, pszTmp))
1248 {
1249 pEntry->fNeedCompiling = 1;
1250 kObjCacheVerbose(pEntry, "object name changed '%s' -> '%s'\n", pEntry->pszObjName, pszTmp);
1251 }
1252 free(pszTmp);
1253 }
1254
1255 /*
1256 * Does the compile command differ?
1257 * TODO: Ignore irrelevant options here (like warning level).
1258 */
1259 if ( !pEntry->fNeedCompiling
1260 && pEntry->cArgvCompile != cArgvCompile)
1261 {
1262 pEntry->fNeedCompiling = 1;
1263 kObjCacheVerbose(pEntry, "compile argument count changed\n");
1264 }
1265 if (!pEntry->fNeedCompiling)
1266 {
1267 unsigned i;
1268 for (i = 0; i < cArgvCompile; i++)
1269 if (strcmp(papszArgvCompile[i], pEntry->papszArgvCompile[i]))
1270 {
1271 pEntry->fNeedCompiling = 1;
1272 kObjCacheVerbose(pEntry, "compile argument differs (%#d)\n", i);
1273 break;
1274 }
1275 }
1276
1277 /*
1278 * Does the object file exist?
1279 */
1280 if ( !pEntry->fNeedCompiling
1281 && !DoesFileInDirExist(pEntry->pszObjName, pEntry->pszDir))
1282 {
1283 pEntry->fNeedCompiling = 1;
1284 kObjCacheVerbose(pEntry, "object file doesn't exist\n");
1285 }
1286
1287 /*
1288 * Does the precompiled output differ in any significant way?
1289 */
1290 if (!pEntry->fNeedCompiling)
1291 {
1292 int fFound = 0;
1293 PCKOCSUM pSum;
1294 for (pSum = &pEntry->SumHead; pSum; pSum = pSum->pNext)
1295 if (kObjCacheSumIsEqual(pSum, &pEntry->NewSum))
1296 {
1297 fFound = 1;
1298 break;
1299 }
1300 if (!fFound)
1301 {
1302 kObjCacheVerbose(pEntry, "no checksum match - comparing output\n");
1303 if (!kObjCacheCompareOldAndNewOutput(pEntry))
1304 pEntry->fNeedCompiling = 1;
1305 else
1306 {
1307 /* insert the sum into the list. */
1308 pEntry->NewSum.pNext = pEntry->SumHead.pNext;
1309 pEntry->SumHead.pNext = &pEntry->NewSum;
1310 }
1311 }
1312 }
1313
1314 /*
1315 * Discard the old precompiled output if it's no longer needed.
1316 */
1317 if ( pEntry->pszOldCppName
1318 && ( !pEntry->fPiped
1319 || pEntry->fNeedCompiling))
1320 {
1321 UnlinkFileInDir(pEntry->pszOldCppName, pEntry->pszDir);
1322 free(pEntry->pszOldCppName);
1323 pEntry->pszOldCppName = NULL;
1324 }
1325
1326 /*
1327 * Do the compliation if found necessary.
1328 */
1329 if (pEntry->fNeedCompiling)
1330 kObjCacheCompileIt(pEntry, papszArgvCompile, cArgvCompile, pszObjName);
1331}
1332
1333
1334/**
1335 * Gets the absolute path
1336 *
1337 * @returns A new heap buffer containing the absolute path.
1338 * @param pszPath The path to make absolute. (Readonly)
1339 */
1340static char *AbsPath(const char *pszPath)
1341{
1342 char szTmp[PATH_MAX];
1343#if defined(__OS2__) || defined(__WIN__)
1344 if (!_fullpath(szTmp, *pszPath ? pszPath : ".", sizeof(szTmp)))
1345 return xstrdup(pszPath);
1346#else
1347 if (!realpath(pszPath, szTmp))
1348 return xstrdup(pszPath);
1349#endif
1350 return xstrdup(szTmp);
1351}
1352
1353
1354/**
1355 * Utility function that finds the filename part in a path.
1356 *
1357 * @returns Pointer to the file name part (this may be "").
1358 * @param pszPath The path to parse.
1359 */
1360static const char *FindFilenameInPath(const char *pszPath)
1361{
1362 const char *pszFilename = strchr(pszPath, '\0') - 1;
1363 while ( pszFilename > pszPath
1364 && !IS_SLASH_DRV(pszFilename[-1]))
1365 pszFilename--;
1366 return pszFilename;
1367}
1368
1369
1370/**
1371 * Utility function that combines a filename and a directory into a path.
1372 *
1373 * @returns malloced buffer containing the result.
1374 * @param pszName The file name.
1375 * @param pszDir The directory path.
1376 */
1377static char *MakePathFromDirAndFile(const char *pszName, const char *pszDir)
1378{
1379 size_t cchName = strlen(pszName);
1380 size_t cchDir = strlen(pszDir);
1381 char *pszBuf = xmalloc(cchName + cchDir + 2);
1382 memcpy(pszBuf, pszDir, cchDir);
1383 if (cchDir > 0 && !IS_SLASH_DRV(pszDir[cchDir - 1]))
1384 pszBuf[cchDir++] = PATH_SLASH;
1385 memcpy(pszBuf + cchDir, pszName, cchName + 1);
1386 return pszBuf;
1387}
1388
1389
1390/**
1391 * Compares two path strings to see if they are identical.
1392 *
1393 * This doesn't do anything fancy, just the case ignoring and
1394 * slash unification.
1395 *
1396 * @returns 1 if equal, 0 otherwise.
1397 * @param pszPath1 The first path.
1398 * @param pszPath2 The second path.
1399 * @param cch The number of characters to compare.
1400 */
1401static int ArePathsIdentical(const char *pszPath1, const char *pszPath2, size_t cch)
1402{
1403#if defined(__OS2__) || defined(__WIN__)
1404 if (strnicmp(pszPath1, pszPath2, cch))
1405 {
1406 /* Slashes may differ, compare char by char. */
1407 const char *psz1 = pszPath1;
1408 const char *psz2 = pszPath2;
1409 for (;cch; psz1++, psz2++, cch--)
1410 {
1411 if (*psz1 != *psz2)
1412 {
1413 if ( tolower(*psz1) != tolower(*psz2)
1414 && toupper(*psz1) != toupper(*psz2)
1415 && *psz1 != '/'
1416 && *psz1 != '\\'
1417 && *psz2 != '/'
1418 && *psz2 != '\\')
1419 return 0;
1420 }
1421 }
1422 }
1423 return 1;
1424#else
1425 return !strncmp(pszPath1, pszPath2, cch);
1426#endif
1427}
1428
1429
1430/**
1431 * Calculate how to get to pszPath from pszDir.
1432 *
1433 * @returns The relative path from pszDir to path pszPath.
1434 * @param pszPath The path to the object.
1435 * @param pszDir The directory it shall be relative to.
1436 */
1437static char *CalcRelativeName(const char *pszPath, const char *pszDir)
1438{
1439 char *pszRet = NULL;
1440 char *pszAbsPath = NULL;
1441 size_t cchDir = strlen(pszDir);
1442
1443 /*
1444 * This is indeed a bit tricky, so we'll try the easy way first...
1445 */
1446 if (ArePathsIdentical(pszPath, pszDir, cchDir))
1447 {
1448 if (pszPath[cchDir])
1449 pszRet = (char *)pszPath + cchDir;
1450 else
1451 pszRet = "./";
1452 }
1453 else
1454 {
1455 pszAbsPath = AbsPath(pszPath);
1456 if (ArePathsIdentical(pszAbsPath, pszDir, cchDir))
1457 {
1458 if (pszPath[cchDir])
1459 pszRet = pszAbsPath + cchDir;
1460 else
1461 pszRet = "./";
1462 }
1463 }
1464 if (pszRet)
1465 {
1466 while (IS_SLASH_DRV(*pszRet))
1467 pszRet++;
1468 pszRet = xstrdup(pszRet);
1469 free(pszAbsPath);
1470 return pszRet;
1471 }
1472
1473 /*
1474 * Damn, it's gonna be complicated. Deal with that later.
1475 */
1476 fprintf(stderr, "kObjCache: complicated relative path stuff isn't implemented yet. sorry.\n");
1477 exit(1);
1478 return NULL;
1479}
1480
1481
1482/**
1483 * Utility function that combines a filename and directory and passes it onto fopen.
1484 *
1485 * @returns fopen return value.
1486 * @param pszName The file name.
1487 * @param pszDir The directory path.
1488 * @param pszMode The fopen mode string.
1489 */
1490static FILE *FOpenFileInDir(const char *pszName, const char *pszDir, const char *pszMode)
1491{
1492 char *pszPath = MakePathFromDirAndFile(pszName, pszDir);
1493 FILE *pFile = fopen(pszPath, pszMode);
1494 free(pszPath);
1495 return pFile;
1496}
1497
1498
1499/**
1500 * Deletes a file in a directory.
1501 *
1502 * @returns whatever unlink returns.
1503 * @param pszName The file name.
1504 * @param pszDir The directory path.
1505 */
1506static int UnlinkFileInDir(const char *pszName, const char *pszDir)
1507{
1508 char *pszPath = MakePathFromDirAndFile(pszName, pszDir);
1509 int rc = unlink(pszPath);
1510 free(pszPath);
1511 return rc;
1512}
1513
1514
1515/**
1516 * Renames a file in a directory.
1517 *
1518 * @returns whatever unlink returns.
1519 * @param pszOldName The new file name.
1520 * @param pszNewName The old file name.
1521 * @param pszDir The directory path.
1522 */
1523static int RenameFileInDir(const char *pszOldName, const char *pszNewName, const char *pszDir)
1524{
1525 char *pszOldPath = MakePathFromDirAndFile(pszOldName, pszDir);
1526 char *pszNewPath = MakePathFromDirAndFile(pszNewName, pszDir);
1527 int rc = rename(pszOldPath, pszNewPath);
1528 free(pszOldPath);
1529 free(pszNewPath);
1530 return rc;
1531}
1532
1533
1534/**
1535 * Check if a (regular) file exists in a directory.
1536 *
1537 * @returns 1 if it exists and is a regular file, 0 if not.
1538 * @param pszName The file name.
1539 * @param pszDir The directory path.
1540 */
1541static int DoesFileInDirExist(const char *pszName, const char *pszDir)
1542{
1543 char *pszPath = MakePathFromDirAndFile(pszName, pszDir);
1544 struct stat st;
1545 int rc = stat(pszPath, &st);
1546 free(pszPath);
1547#ifdef S_ISREG
1548 return !rc && S_ISREG(st.st_mode);
1549#elif defined(_MSC_VER)
1550 return !rc && (st.st_mode & _S_IFMT) == _S_IFREG;
1551#else
1552#error "Port me"
1553#endif
1554}
1555
1556
1557/**
1558 * Reads into memory an entire file.
1559 *
1560 * @returns Pointer to the heap allocation containing the file.
1561 * On failure NULL and errno is returned.
1562 * @param pszName The file.
1563 * @param pszDir The directory the file resides in.
1564 * @param pcbFile Where to store the file size.
1565 */
1566static void *ReadFileInDir(const char *pszName, const char *pszDir, size_t *pcbFile)
1567{
1568 int SavedErrno;
1569 char *pszPath = MakePathFromDirAndFile(pszName, pszDir);
1570 int fd = open(pszPath, O_RDONLY | O_BINARY);
1571 if (fd >= 0)
1572 {
1573 off_t cbFile = lseek(fd, 0, SEEK_END);
1574 if ( cbFile >= 0
1575 && lseek(fd, 0, SEEK_SET) == 0)
1576 {
1577 char *pb = malloc(cbFile + 1);
1578 if (pb)
1579 {
1580 if (read(fd, pb, cbFile) == cbFile)
1581 {
1582 close(fd);
1583 pb[cbFile] = '\0';
1584 *pcbFile = (size_t)cbFile;
1585 return pb;
1586 }
1587 SavedErrno = errno;
1588 free(pb);
1589 }
1590 else
1591 SavedErrno = ENOMEM;
1592 }
1593 else
1594 SavedErrno = errno;
1595 close(fd);
1596 }
1597 else
1598 SavedErrno = errno;
1599 free(pszPath);
1600 errno = SavedErrno;
1601 return NULL;
1602}
1603
1604
1605static void *xmalloc(size_t cb)
1606{
1607 void *pv = malloc(cb);
1608 if (!pv)
1609 kObjCacheFatal(NULL, "out of memory (%d)\n", (int)cb);
1610 return pv;
1611}
1612
1613
1614static void *xrealloc(void *pvOld, size_t cb)
1615{
1616 void *pv = realloc(pvOld, cb);
1617 if (!pv)
1618 kObjCacheFatal(NULL, "out of memory (%d)\n", (int)cb);
1619 return pv;
1620}
1621
1622
1623static char *xstrdup(const char *pszIn)
1624{
1625 char *psz = strdup(pszIn);
1626 if (!psz)
1627 kObjCacheFatal(NULL, "out of memory (%d)\n", (int)strlen(pszIn));
1628 return psz;
1629}
1630
1631
1632/**
1633 * Prints a syntax error and returns the appropriate exit code
1634 *
1635 * @returns approriate exit code.
1636 * @param pszFormat The syntax error message.
1637 * @param ... Message args.
1638 */
1639static int SyntaxError(const char *pszFormat, ...)
1640{
1641 va_list va;
1642 fprintf(stderr, "kObjCache: syntax error: ");
1643 va_start(va, pszFormat);
1644 vfprintf(stderr, pszFormat, va);
1645 va_end(va);
1646 return 1;
1647}
1648
1649
1650/**
1651 * Prints the usage.
1652 * @returns 0.
1653 */
1654static int usage(void)
1655{
1656 printf("syntax: kObjCache [-v|--verbose] [-f|--file] <cache-file> [-V|--version] [-r|--redir-stdout]\n"
1657 " --kObjCache-cpp <filename> <precompiler + args> \n"
1658 " --kObjCache-cc <object> <compiler + args>\n"
1659 " [--kObjCache-both [args]]\n"
1660 " [--kObjCache-cpp|--kObjCache-cc [more args]]\n"
1661 "\n");
1662 return 0;
1663}
1664
1665
1666int main(int argc, char **argv)
1667{
1668 PKOBJCACHE pEntry;
1669
1670 const char *pszCacheFile;
1671
1672 const char **papszArgvPreComp = NULL;
1673 unsigned cArgvPreComp = 0;
1674 const char *pszPreCompName = NULL;
1675 int fRedirStdOut = 0;
1676
1677 const char **papszArgvCompile = NULL;
1678 unsigned cArgvCompile = 0;
1679 const char *pszObjName = NULL;
1680
1681 enum { kOC_Options, kOC_CppArgv, kOC_CcArgv, kOC_BothArgv } enmMode = kOC_Options;
1682
1683 int i;
1684
1685 /*
1686 * Parse the arguments.
1687 */
1688 for (i = 1; i < argc; i++)
1689 {
1690 if (!strcmp(argv[i], "--kObjCache-cpp"))
1691 {
1692 enmMode = kOC_CppArgv;
1693 if (!pszPreCompName)
1694 {
1695 if (++i >= argc)
1696 return SyntaxError("--kObjCache-cpp requires an object filename!\n");
1697 pszPreCompName = argv[i];
1698 }
1699 }
1700 else if (!strcmp(argv[i], "--kObjCache-cc"))
1701 {
1702 enmMode = kOC_CcArgv;
1703 if (!pszObjName)
1704 {
1705 if (++i >= argc)
1706 return SyntaxError("--kObjCache-cc requires an precompiler output filename!\n");
1707 pszObjName = argv[i];
1708 }
1709 }
1710 else if (!strcmp(argv[i], "--kObjCache-both"))
1711 enmMode = kOC_BothArgv;
1712 else if (!strcmp(argv[i], "--help"))
1713 return usage();
1714 else if (enmMode != kOC_Options)
1715 {
1716 if (enmMode == kOC_CppArgv || enmMode == kOC_BothArgv)
1717 {
1718 if (!(cArgvPreComp % 16))
1719 papszArgvPreComp = xrealloc((void *)papszArgvPreComp, (cArgvPreComp + 17) * sizeof(papszArgvPreComp[0]));
1720 papszArgvPreComp[cArgvPreComp++] = argv[i];
1721 papszArgvPreComp[cArgvPreComp] = NULL;
1722 }
1723 if (enmMode == kOC_CcArgv || enmMode == kOC_BothArgv)
1724 {
1725 if (!(cArgvCompile % 16))
1726 papszArgvCompile = xrealloc((void *)papszArgvCompile, (cArgvCompile + 17) * sizeof(papszArgvCompile[0]));
1727 papszArgvCompile[cArgvCompile++] = argv[i];
1728 papszArgvCompile[cArgvCompile] = NULL;
1729 }
1730 }
1731 else if (!strcmp(argv[i], "-f") || !strcmp(argv[i], "--file"))
1732 {
1733 if (i + 1 >= argc)
1734 return SyntaxError("%s requires a cache filename!\n", argv[i]);
1735 pszCacheFile = argv[++i];
1736 }
1737 else if (!strcmp(argv[i], "-r") || !strcmp(argv[i], "--redir-stdout"))
1738 fRedirStdOut = 1;
1739 else if (!strcmp(argv[i], "-v") || !strcmp(argv[i], "--verbose"))
1740 g_fVerbose = 1;
1741 else if (!strcmp(argv[i], "-q") || !strcmp(argv[i], "--quiet"))
1742 g_fVerbose = 0;
1743 else if (!strcmp(argv[i], "-h") || !strcmp(argv[i], "-?"))
1744 return usage();
1745 else if (!strcmp(argv[i], "-V") || !strcmp(argv[i], "--version"))
1746 {
1747 printf("kObjCache v0.0.0 ($Revision: 1036 $)\n");
1748 return 0;
1749 }
1750 else
1751 return SyntaxError("Doesn't grok '%s'!\n", argv[i]);
1752 }
1753 if (!pszCacheFile)
1754 return SyntaxError("No cache file name (-f)\n");
1755 if (!cArgvCompile)
1756 return SyntaxError("No compiler arguments (--kObjCache-cc)\n");
1757 if (!cArgvPreComp)
1758 return SyntaxError("No precompiler arguments (--kObjCache-cc)\n");
1759
1760 /*
1761 * Create a cache entry from the cache file (if found).
1762 */
1763 pEntry = kObjCacheCreate(pszCacheFile);
1764 kObjCacheRead(pEntry);
1765
1766 /*
1767 * Do the compiling.
1768 */
1769 kObjCachePreCompile(pEntry, papszArgvPreComp, cArgvPreComp, pszPreCompName, fRedirStdOut);
1770 kObjCacheCompileIfNeeded(pEntry, papszArgvCompile, cArgvCompile, pszObjName);
1771
1772 /*
1773 * Write the cache file.
1774 */
1775 kObjCacheWrite(pEntry);
1776 /* kObjCacheDestroy(pEntry); - don't bother */
1777 return 0;
1778}
1779
1780
1781/** @page kObjCache Benchmarks.
1782 *
1783 * 2007-06-02 - 21-23:00:
1784 * Mac OS X debug -j 3 clobber build (rm -Rf out/darwin.x86/debug ; sync ; svn diff ; sync ; sleep 1 ; time kmk -j 3):
1785 * real 10m26.077s
1786 * user 13m13.291s
1787 * sys 2m58.193s
1788 *
1789 * Mac OS X debug -j 3 depend build (touch include/iprt/err.h ; sync ; svn diff ; sync ; sleep 1 ; time kmk -j 3):
1790 * real 3m55.275s
1791 * user 4m11.852s
1792 * sys 0m54.931s
1793 *
1794 * Mac OS X debug -j 3 cached clobber build (rm -Rf out/darwin.x86/debug ; sync ; svn diff ; sync ; sleep 1 ; time kmk -j 3 USE_KOBJCACHE=1):
1795 * real 11m42.513s
1796 * user 14m27.736s
1797 * sys 3m39.512s
1798 *
1799 * Mac OS X debug -j 3 cached depend build (touch include/iprt/err.h ; sync ; svn diff ; sync ; sleep 1 ; time kmk -j 3 USE_KOBJCACHE=1):
1800 * real 1m17.445s
1801 * user 1m13.410s
1802 * sys 0m22.789s
1803 *
1804 * Mac OS X debug -j3 cached depend build (touch include/iprt/cdefs.h ; sync ; svn diff ; sync ; sleep 1 ; time kmk -j 3 USE_KOBJCACHE=1):
1805 * real 1m29.315s
1806 * user 1m31.391s
1807 * sys 0m32.748s
1808 *
1809 */
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette