/***************************************************************************** * This file is part of Kvazaar HEVC encoder. * * Copyright (C) 2013-2015 Tampere University of Technology and others (see * COPYING file). * * Kvazaar is free software: you can redistribute it and/or modify it under * the terms of the GNU Lesser General Public License as published by the * Free Software Foundation; either version 2.1 of the License, or (at your * option) any later version. * * Kvazaar is distributed in the hope that it will be useful, but WITHOUT ANY * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS * FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for * more details. * * You should have received a copy of the GNU General Public License along * with Kvazaar. If not, see . ****************************************************************************/ /* * \file */ #include "strategyselector.h" #include #include #include #if COMPILE_INTEL #include #endif hardware_flags_t g_hardware_flags; static void set_hardware_flags(int32_t cpuid); static void* strategyselector_choose_for(const strategy_list_t * const strategies, const char * const strategy_type); //Strategies to include (add new file here) //Returns 1 if successful int strategyselector_init(int32_t cpuid) { const strategy_to_select_t *cur_strategy_to_select = strategies_to_select; strategy_list_t strategies; strategies.allocated = 0; strategies.count = 0; strategies.strategies = NULL; set_hardware_flags(cpuid); //Add new register function here if (!strategy_register_picture(&strategies)) { fprintf(stderr, "strategy_register_picture failed!\n"); return 0; } if (!strategy_register_nal(&strategies)) { fprintf(stderr, "strategy_register_nal failed!\n"); return 0; } if (!strategy_register_dct(&strategies)) { fprintf(stderr, "strategy_register_dct failed!\n"); return 0; } if (!strategy_register_ipol(&strategies)) { fprintf(stderr, "strategy_register_ipol failed!\n"); return 0; } while(cur_strategy_to_select->fptr) { *(cur_strategy_to_select->fptr) = strategyselector_choose_for(&strategies, cur_strategy_to_select->strategy_type); if (!(*(cur_strategy_to_select->fptr))) { fprintf(stderr, "Could not find a strategy for %s!\n", cur_strategy_to_select->strategy_type); return 0; } ++cur_strategy_to_select; } //We can free the structure now, as all strategies are statically set to pointers if (strategies.allocated) { free(strategies.strategies); } return 1; } //Returns 1 if successful, 0 otherwise int strategyselector_register(void * const opaque, const char * const type, const char * const strategy_name, int priority, void * const fptr) { strategy_list_t * const strategies = opaque; if (strategies->allocated == strategies->count) { strategy_t* new_strategies = realloc(strategies->strategies, sizeof(strategy_t) * (strategies->allocated + STRATEGY_LIST_ALLOC_SIZE)); if (!new_strategies) { fprintf(stderr, "Could not increase strategies list size!\n"); return 0; } strategies->strategies = new_strategies; strategies->allocated += STRATEGY_LIST_ALLOC_SIZE; } { strategy_t *new_strategy = &strategies->strategies[strategies->count++]; new_strategy->type = type; new_strategy->strategy_name = strategy_name; new_strategy->priority = priority; new_strategy->fptr = fptr; } #ifdef DEBUG_STRATEGYSELECTOR fprintf(stderr, "Registered strategy %s:%s with priority %d (%p)\n", type, strategy_name, priority, fptr); #endif //DEBUG_STRATEGYSELECTOR return 1; } static void* strategyselector_choose_for(const strategy_list_t * const strategies, const char * const strategy_type) { unsigned int max_priority = 0; int max_priority_i = -1; char buffer[256]; char *override = NULL; int i = 0; // Because VS doesn't support snprintf, let's assert that there is // enough room in the buffer. Max length for strategy type is // buffersize (256) - prefix including terminating zero. assert(strlen(strategy_type) < 256 - sizeof("KVAZAAR_OVERRIDE_") ); sprintf(buffer, "KVAZAAR_OVERRIDE_%s", strategy_type); override = getenv(buffer); for (i=0; i < strategies->count; ++i) { if (strcmp(strategies->strategies[i].type, strategy_type) == 0) { if (override && strcmp(strategies->strategies[i].strategy_name, override) == 0) { fprintf(stderr, "%s environment variable present, choosing %s:%s\n", buffer, strategy_type, strategies->strategies[i].strategy_name); return strategies->strategies[i].fptr; } if (strategies->strategies[i].priority >= max_priority) { max_priority_i = i; max_priority = strategies->strategies[i].priority; } } } if (override) { fprintf(stderr, "%s environment variable present, but no strategy %s was found!\n", buffer, override); return NULL; } #ifdef DEBUG_STRATEGYSELECTOR fprintf(stderr, "Choosing strategy for %s:\n", strategy_type); for (i=0; i < strategies->count; ++i) { if (strcmp(strategies->strategies[i].type, strategy_type) == 0) { if (i != max_priority_i) { fprintf(stderr, "- %s (%d, %p)\n", strategies->strategies[i].strategy_name, strategies->strategies[i].priority, strategies->strategies[i].fptr); } else { fprintf(stderr, "> %s (%d, %p)\n", strategies->strategies[i].strategy_name, strategies->strategies[i].priority, strategies->strategies[i].fptr); } } } #endif //DEBUG_STRATEGYSELECTOR if (max_priority_i == -1) { return NULL; } return strategies->strategies[max_priority_i].fptr; } #if COMPILE_INTEL #if defined(__GNUC__) #include static INLINE int get_cpuid(unsigned int level, unsigned int *eax, unsigned int *ebx, unsigned int *ecx, unsigned int *edx) { return __get_cpuid(level, eax, ebx, ecx, edx); } #else #include //Adapter from __cpuid (VS) to __get_cpuid (GNU C). static INLINE int get_cpuid(unsigned int level, unsigned int *eax, unsigned int *ebx, unsigned int *ecx, unsigned int *edx) { int CPUInfo[4] = {*eax, *ebx, *ecx, *edx}; __cpuid(CPUInfo, 0); // check if the CPU supports the cpuid instruction. if (CPUInfo[0] != 0) { __cpuid(CPUInfo, level); *eax = CPUInfo[0]; *ebx = CPUInfo[1]; *ecx = CPUInfo[2]; *edx = CPUInfo[3]; return 1; } return 0; } #endif //defined(__GNUC__) #endif #if COMPILE_POWERPC #include #include #include #include //Source: http://freevec.org/function/altivec_runtime_detection_linux static int altivec_available(void) { int result = 0; unsigned long buf[64]; ssize_t count; int fd, i; fd = open("/proc/self/auxv", O_RDONLY); if (fd < 0) { return 0; } // loop on reading do { count = read(fd, buf, sizeof(buf)); if (count < 0) break; for (i=0; i < (count / sizeof(unsigned long)); i += 2) { if (buf[i] == AT_HWCAP) { result = !!(buf[i+1] & PPC_FEATURE_HAS_ALTIVEC); goto out_close; } else if (buf[i] == AT_NULL) goto out_close; } } while (count == sizeof(buf)); out_close: close(fd); return result; } #endif //COMPILE_POWERPC static void set_hardware_flags(int32_t cpuid) { FILL(g_hardware_flags, 0); g_hardware_flags.arm = COMPILE_ARM; g_hardware_flags.intel = COMPILE_INTEL; g_hardware_flags.powerpc = COMPILE_POWERPC; #if COMPILE_INTEL if (cpuid) { unsigned int eax = 0, ebx = 0, ecx = 0, edx =0; /* CPU feature bits */ enum { BIT_SSE3 = 0, BIT_SSSE3 = 9, BIT_SSE41 = 19, BIT_SSE42 = 20, BIT_MMX = 24, BIT_SSE = 25, BIT_SSE2 = 26, BIT_OSXSAVE = 27, BIT_AVX = 28}; enum { XCR0_XMM = 1, XCR0_YMM = 2 }; // Dig CPU features with cpuid get_cpuid(1, &eax, &ebx, &ecx, &edx); // EDX if (edx & (1<