| /* |
| * Copyright (C) 2010 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| #include <memory.h> |
| #include <setjmp.h> |
| #include <stdio.h> |
| #include <signal.h> |
| |
| #include <cpu-features.h> |
| |
| #ifndef __arm__ |
| #error "Only compile this file for an ARM target" |
| #endif |
| |
| static int volatile g_signal_raised; |
| static sigjmp_buf g_jumper; |
| |
| static void my_signal_handler(int signum) { |
| g_signal_raised = 1; |
| siglongjmp(g_jumper, 1); |
| } |
| |
| static int32_t do_idiv(int32_t a, int32_t b) { |
| __asm__ __volatile__( |
| "sdiv %0, %0, %1\n\t" |
| : "=&r"(a) |
| : "r"(b) |
| : "cc"); |
| return a; |
| } |
| |
| // Check that the ARM idiv instruction is supported. |
| // Returns 1 on success, 0 otherwise. |
| static int check_idiv(void) { |
| // Setup SIGILL signal handler. |
| struct sigaction old_handler, new_handler; |
| memset(&new_handler, 0, sizeof(new_handler)); |
| new_handler.sa_handler = my_signal_handler; |
| sigaction(SIGILL, &new_handler, &old_handler); |
| |
| // Do the division. |
| g_signal_raised = 0; |
| if (sigsetjmp(g_jumper, 0) == 0) |
| do_idiv(12345, 17); |
| |
| // Restore SIGILL handler. |
| sigaction(SIGILL, &old_handler, NULL); |
| |
| return g_signal_raised == 0; |
| } |
| |
| int main(void) |
| { |
| uint32_t cpu_id = android_getCpuIdArm(); |
| printf("cpu-features reports the following CPUID: 0x%08x\n", |
| cpu_id); |
| |
| uint64_t features = android_getCpuFeatures(); |
| #ifdef __thumb2__ |
| const uint64_t idiv_flag = ANDROID_CPU_ARM_FEATURE_IDIV_THUMB2; |
| const char* variant = "thumb"; |
| #else |
| const uint64_t idiv_flag = ANDROID_CPU_ARM_FEATURE_IDIV_ARM; |
| const char* variant = "arm"; |
| #endif |
| |
| printf("status of %s idiv instruction:\n", variant); |
| int idiv_reported = (features & idiv_flag) != 0; |
| printf(" reported : %s\n", |
| idiv_reported ? "supported" : "unsupported"); |
| |
| int idiv_checked = check_idiv(); |
| printf(" runtime check : %s\n", |
| idiv_checked ? "supported" : "unsupported"); |
| |
| return (idiv_reported != idiv_checked); |
| } |