mirror of
				https://github.com/ruby/ruby.git
				synced 2022-11-09 12:17:21 -05:00 
			
		
		
		
	According to MSVC manual (*1), cl.exe can skip including a header file when that: - contains #pragma once, or - starts with #ifndef, or - starts with #if ! defined. GCC has a similar trick (*2), but it acts more stricter (e. g. there must be _no tokens_ outside of #ifndef...#endif). Sun C lacked #pragma once for a looong time. Oracle Developer Studio 12.5 finally implemented it, but we cannot assume such recent version. This changeset modifies header files so that each of them include strictly one #ifndef...#endif. I believe this is the most portable way to trigger compiler optimizations. [Bug #16770] *1: https://docs.microsoft.com/en-us/cpp/preprocessor/once *2: https://gcc.gnu.org/onlinedocs/cppinternals/Guard-Macros.html
		
			
				
	
	
		
			50 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			50 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#ifndef INTERNAL_COMPAR_H                                /*-*-C-*-vi:se ft=c:*/
 | 
						|
#define INTERNAL_COMPAR_H
 | 
						|
/**
 | 
						|
 * @file
 | 
						|
 * @author     Ruby developers <ruby-core@ruby-lang.org>
 | 
						|
 * @copyright  This  file  is   a  part  of  the   programming  language  Ruby.
 | 
						|
 *             Permission  is hereby  granted,  to  either redistribute  and/or
 | 
						|
 *             modify this file, provided that  the conditions mentioned in the
 | 
						|
 *             file COPYING are met.  Consult the file for details.
 | 
						|
 * @brief      Internal header for Comparable.
 | 
						|
 */
 | 
						|
#include "internal/vm.h"        /* for rb_method_basic_definition_p */
 | 
						|
 | 
						|
#define STRING_P(s) (RB_TYPE_P((s), T_STRING) && CLASS_OF(s) == rb_cString)
 | 
						|
 | 
						|
enum {
 | 
						|
    cmp_opt_Integer,
 | 
						|
    cmp_opt_String,
 | 
						|
    cmp_opt_Float,
 | 
						|
    cmp_optimizable_count
 | 
						|
};
 | 
						|
 | 
						|
struct cmp_opt_data {
 | 
						|
    unsigned int opt_methods;
 | 
						|
    unsigned int opt_inited;
 | 
						|
};
 | 
						|
 | 
						|
#define NEW_CMP_OPT_MEMO(type, value) \
 | 
						|
    NEW_PARTIAL_MEMO_FOR(type, value, cmp_opt)
 | 
						|
#define CMP_OPTIMIZABLE_BIT(type) (1U << TOKEN_PASTE(cmp_opt_,type))
 | 
						|
#define CMP_OPTIMIZABLE(data, type) \
 | 
						|
    (((data).opt_inited & CMP_OPTIMIZABLE_BIT(type)) ? \
 | 
						|
     ((data).opt_methods & CMP_OPTIMIZABLE_BIT(type)) : \
 | 
						|
     (((data).opt_inited |= CMP_OPTIMIZABLE_BIT(type)), \
 | 
						|
      rb_method_basic_definition_p(TOKEN_PASTE(rb_c,type), id_cmp) && \
 | 
						|
      ((data).opt_methods |= CMP_OPTIMIZABLE_BIT(type))))
 | 
						|
 | 
						|
#define OPTIMIZED_CMP(a, b, data) \
 | 
						|
    ((FIXNUM_P(a) && FIXNUM_P(b) && CMP_OPTIMIZABLE(data, Integer)) ? \
 | 
						|
     (((long)a > (long)b) ? 1 : ((long)a < (long)b) ? -1 : 0) : \
 | 
						|
     (STRING_P(a) && STRING_P(b) && CMP_OPTIMIZABLE(data, String)) ? \
 | 
						|
     rb_str_cmp(a, b) : \
 | 
						|
     (RB_FLOAT_TYPE_P(a) && RB_FLOAT_TYPE_P(b) && CMP_OPTIMIZABLE(data, Float)) ? \
 | 
						|
     rb_float_cmp(a, b) : \
 | 
						|
     rb_cmpint(rb_funcallv(a, id_cmp, 1, &b), a, b))
 | 
						|
 | 
						|
/* compar.c */
 | 
						|
VALUE rb_invcmp(VALUE, VALUE);
 | 
						|
 | 
						|
#endif /* INTERNAL_COMPAR_H */
 |