19
19
#include "split-index.h"
20
20
#include "utf8.h"
21
21
22
+ #ifndef NO_PTHREADS
23
+ #include <pthread.h>
24
+ #endif
25
+
22
26
/* Mask for the name length in ce_flags in the on-disk index */
23
27
24
28
#define CE_NAMEMASK (0x0fff)
@@ -1391,6 +1395,34 @@ static int verify_hdr(struct cache_header *hdr, unsigned long size)
1391
1395
return 0 ;
1392
1396
}
1393
1397
1398
+ #ifndef NO_PTHREADS
1399
+ /*
1400
+ * Require index file to be larger than this threshold before
1401
+ * we bother using a thread to verify the SHA.
1402
+ * This value was arbitrarily chosen.
1403
+ */
1404
+ #define VERIFY_HDR_THRESHOLD 10*1024*1024
1405
+
1406
+ struct verify_hdr_thread_data
1407
+ {
1408
+ pthread_t thread_id ;
1409
+ struct cache_header * hdr ;
1410
+ size_t size ;
1411
+ int result ;
1412
+ };
1413
+
1414
+ /*
1415
+ * A thread proc to run the verify_hdr() computation
1416
+ * in a background thread.
1417
+ */
1418
+ static void * verify_hdr_thread (void * _data )
1419
+ {
1420
+ struct verify_hdr_thread_data * p = _data ;
1421
+ p -> result = verify_hdr (p -> hdr , (unsigned long )p -> size );
1422
+ return NULL ;
1423
+ }
1424
+ #endif
1425
+
1394
1426
static int read_index_extension (struct index_state * istate ,
1395
1427
const char * ext , void * data , unsigned long sz )
1396
1428
{
@@ -1575,6 +1607,9 @@ int do_read_index(struct index_state *istate, const char *path, int must_exist)
1575
1607
void * mmap ;
1576
1608
size_t mmap_size ;
1577
1609
struct strbuf previous_name_buf = STRBUF_INIT , * previous_name ;
1610
+ #ifndef NO_PTHREADS
1611
+ struct verify_hdr_thread_data verify_hdr_thread_data ;
1612
+ #endif
1578
1613
1579
1614
if (istate -> initialized )
1580
1615
return istate -> cache_nr ;
@@ -1601,8 +1636,23 @@ int do_read_index(struct index_state *istate, const char *path, int must_exist)
1601
1636
close (fd );
1602
1637
1603
1638
hdr = mmap ;
1639
+ #ifdef NO_PTHREADS
1604
1640
if (verify_hdr (hdr , mmap_size ) < 0 )
1605
1641
goto unmap ;
1642
+ #else
1643
+ if (mmap_size < VERIFY_HDR_THRESHOLD ) {
1644
+ if (verify_hdr (hdr , mmap_size ) < 0 )
1645
+ goto unmap ;
1646
+ } else {
1647
+ verify_hdr_thread_data .hdr = hdr ;
1648
+ verify_hdr_thread_data .size = mmap_size ;
1649
+ verify_hdr_thread_data .result = -1 ;
1650
+ if (pthread_create (
1651
+ & verify_hdr_thread_data .thread_id , NULL ,
1652
+ verify_hdr_thread , & verify_hdr_thread_data ))
1653
+ die_errno ("unable to start verify_hdr_thread" );
1654
+ }
1655
+ #endif
1606
1656
1607
1657
hashcpy (istate -> sha1 , (const unsigned char * )hdr + mmap_size - 20 );
1608
1658
istate -> version = ntohl (hdr -> hdr_version );
@@ -1650,6 +1700,16 @@ int do_read_index(struct index_state *istate, const char *path, int must_exist)
1650
1700
src_offset += 8 ;
1651
1701
src_offset += extsize ;
1652
1702
}
1703
+
1704
+ #ifndef NO_PTHREADS
1705
+ if (mmap_size >= VERIFY_HDR_THRESHOLD ) {
1706
+ if (pthread_join (verify_hdr_thread_data .thread_id , NULL ))
1707
+ die_errno ("unable to join verify_hdr_thread" );
1708
+ if (verify_hdr_thread_data .result < 0 )
1709
+ goto unmap ;
1710
+ }
1711
+ #endif
1712
+
1653
1713
munmap (mmap , mmap_size );
1654
1714
return istate -> cache_nr ;
1655
1715
0 commit comments