2006-02-08 23:33:12 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) Igor Sysoev
|
2012-01-18 23:07:43 +08:00
|
|
|
* Copyright (C) Nginx, Inc.
|
2006-02-08 23:33:12 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <ngx_config.h>
|
|
|
|
#include <ngx_core.h>
|
|
|
|
|
|
|
|
|
|
|
|
#if (( __i386__ || __amd64__ ) && ( __GNUC__ || __INTEL_COMPILER ))
|
|
|
|
|
|
|
|
|
|
|
|
static ngx_inline void ngx_cpuid(uint32_t i, uint32_t *buf);
|
|
|
|
|
|
|
|
|
2006-02-23 03:41:39 +08:00
|
|
|
#if ( __i386__ )
|
|
|
|
|
|
|
|
static ngx_inline void
|
|
|
|
ngx_cpuid(uint32_t i, uint32_t *buf)
|
|
|
|
{
|
|
|
|
|
|
|
|
/*
|
|
|
|
* we could not use %ebx as output parameter if gcc builds PIC,
|
|
|
|
* and we could not save %ebx on stack, because %esp is used,
|
|
|
|
* when the -fomit-frame-pointer optimization is specified.
|
|
|
|
*/
|
|
|
|
|
|
|
|
__asm__ (
|
|
|
|
|
|
|
|
" mov %%ebx, %%esi; "
|
|
|
|
|
|
|
|
" cpuid; "
|
2006-05-23 22:54:58 +08:00
|
|
|
" mov %%eax, (%1); "
|
|
|
|
" mov %%ebx, 4(%1); "
|
|
|
|
" mov %%edx, 8(%1); "
|
|
|
|
" mov %%ecx, 12(%1); "
|
2006-02-23 03:41:39 +08:00
|
|
|
|
|
|
|
" mov %%esi, %%ebx; "
|
|
|
|
|
2006-05-23 22:54:58 +08:00
|
|
|
: : "a" (i), "D" (buf) : "ecx", "edx", "esi", "memory" );
|
2006-02-23 03:41:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#else /* __amd64__ */
|
|
|
|
|
|
|
|
|
2006-02-08 23:33:12 +08:00
|
|
|
static ngx_inline void
|
|
|
|
ngx_cpuid(uint32_t i, uint32_t *buf)
|
|
|
|
{
|
|
|
|
uint32_t eax, ebx, ecx, edx;
|
|
|
|
|
|
|
|
__asm__ (
|
|
|
|
|
|
|
|
"cpuid"
|
|
|
|
|
|
|
|
: "=a" (eax), "=b" (ebx), "=c" (ecx), "=d" (edx) : "a" (i) );
|
|
|
|
|
|
|
|
buf[0] = eax;
|
|
|
|
buf[1] = ebx;
|
|
|
|
buf[2] = edx;
|
|
|
|
buf[3] = ecx;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-23 03:41:39 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2006-02-08 23:33:12 +08:00
|
|
|
/* auto detect the L2 cache line size of modern and widespread CPUs */
|
|
|
|
|
|
|
|
void
|
|
|
|
ngx_cpuinfo(void)
|
|
|
|
{
|
|
|
|
u_char *vendor;
|
2009-03-28 20:43:41 +08:00
|
|
|
uint32_t vbuf[5], cpu[4], model;
|
2006-02-08 23:33:12 +08:00
|
|
|
|
|
|
|
vbuf[0] = 0;
|
|
|
|
vbuf[1] = 0;
|
|
|
|
vbuf[2] = 0;
|
|
|
|
vbuf[3] = 0;
|
|
|
|
vbuf[4] = 0;
|
|
|
|
|
|
|
|
ngx_cpuid(0, vbuf);
|
|
|
|
|
|
|
|
vendor = (u_char *) &vbuf[1];
|
|
|
|
|
|
|
|
if (vbuf[0] == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ngx_cpuid(1, cpu);
|
|
|
|
|
|
|
|
if (ngx_strcmp(vendor, "GenuineIntel") == 0) {
|
|
|
|
|
2008-01-29 14:58:47 +08:00
|
|
|
switch ((cpu[0] & 0xf00) >> 8) {
|
2006-02-08 23:33:12 +08:00
|
|
|
|
|
|
|
/* Pentium */
|
|
|
|
case 5:
|
2008-01-29 15:06:18 +08:00
|
|
|
ngx_cacheline_size = 32;
|
|
|
|
break;
|
|
|
|
|
2006-02-08 23:33:12 +08:00
|
|
|
/* Pentium Pro, II, III */
|
|
|
|
case 6:
|
|
|
|
ngx_cacheline_size = 32;
|
2008-01-29 15:06:18 +08:00
|
|
|
|
2009-03-28 20:43:41 +08:00
|
|
|
model = ((cpu[0] & 0xf0000) >> 8) | (cpu[0] & 0xf0);
|
|
|
|
|
|
|
|
if (model >= 0xd0) {
|
|
|
|
/* Intel Core, Core 2, Atom */
|
2008-01-29 15:06:18 +08:00
|
|
|
ngx_cacheline_size = 64;
|
|
|
|
}
|
|
|
|
|
2006-02-08 23:33:12 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Pentium 4, although its cache line size is 64 bytes,
|
|
|
|
* it prefetches up to two cache lines during memory read
|
|
|
|
*/
|
|
|
|
case 15:
|
|
|
|
ngx_cacheline_size = 128;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if (ngx_strcmp(vendor, "AuthenticAMD") == 0) {
|
|
|
|
ngx_cacheline_size = 64;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
ngx_cpuinfo(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#endif
|