2016-05-18 02:52:22 +08:00
|
|
|
/*
|
2017-10-02 17:24:02 +08:00
|
|
|
* Copyright 2010-2017 The OpenSSL Project Authors. All Rights Reserved.
|
2016-05-18 02:52:22 +08:00
|
|
|
*
|
|
|
|
* Licensed under the OpenSSL license (the "License"). You may not use
|
|
|
|
* this file except in compliance with the License. You can obtain a copy
|
|
|
|
* in the file LICENSE in the source distribution or at
|
|
|
|
* https://www.openssl.org/source/license.html
|
|
|
|
*/
|
|
|
|
|
2010-01-19 20:24:59 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <setjmp.h>
|
|
|
|
#include <signal.h>
|
2016-10-15 23:41:41 +08:00
|
|
|
#include "internal/cryptlib.h"
|
2017-10-02 17:24:02 +08:00
|
|
|
#include "s390x_arch.h"
|
2010-01-19 20:24:59 +08:00
|
|
|
|
|
|
|
static sigjmp_buf ill_jmp;
|
2015-01-22 11:40:55 +08:00
|
|
|
static void ill_handler(int sig)
|
|
|
|
{
|
|
|
|
siglongjmp(ill_jmp, sig);
|
|
|
|
}
|
2010-01-19 20:24:59 +08:00
|
|
|
|
2017-10-02 17:24:02 +08:00
|
|
|
void OPENSSL_s390x_facilities(void);
|
|
|
|
void OPENSSL_vx_probe(void);
|
|
|
|
|
|
|
|
struct OPENSSL_s390xcap_st OPENSSL_s390xcap_P;
|
2010-01-19 20:24:59 +08:00
|
|
|
|
|
|
|
void OPENSSL_cpuid_setup(void)
|
2015-01-22 11:40:55 +08:00
|
|
|
{
|
|
|
|
sigset_t oset;
|
|
|
|
struct sigaction ill_act, oact;
|
|
|
|
|
2017-10-02 17:24:02 +08:00
|
|
|
if (OPENSSL_s390xcap_P.stfle[0])
|
2015-01-22 11:40:55 +08:00
|
|
|
return;
|
|
|
|
|
2017-10-02 17:24:02 +08:00
|
|
|
/* set a bit that will not be tested later */
|
|
|
|
OPENSSL_s390xcap_P.stfle[0] |= S390X_CAPBIT(0);
|
2015-01-22 11:40:55 +08:00
|
|
|
|
|
|
|
memset(&ill_act, 0, sizeof(ill_act));
|
|
|
|
ill_act.sa_handler = ill_handler;
|
|
|
|
sigfillset(&ill_act.sa_mask);
|
|
|
|
sigdelset(&ill_act.sa_mask, SIGILL);
|
2017-10-02 17:24:02 +08:00
|
|
|
sigdelset(&ill_act.sa_mask, SIGFPE);
|
2015-01-22 11:40:55 +08:00
|
|
|
sigdelset(&ill_act.sa_mask, SIGTRAP);
|
|
|
|
sigprocmask(SIG_SETMASK, &ill_act.sa_mask, &oset);
|
|
|
|
sigaction(SIGILL, &ill_act, &oact);
|
2017-10-02 17:24:02 +08:00
|
|
|
sigaction(SIGFPE, &ill_act, &oact);
|
2015-01-22 11:40:55 +08:00
|
|
|
|
|
|
|
/* protection against missing store-facility-list-extended */
|
|
|
|
if (sigsetjmp(ill_jmp, 1) == 0)
|
|
|
|
OPENSSL_s390x_facilities();
|
|
|
|
|
2017-10-02 17:24:02 +08:00
|
|
|
/* protection against disabled vector facility */
|
|
|
|
if ((OPENSSL_s390xcap_P.stfle[2] & S390X_CAPBIT(S390X_VX))
|
|
|
|
&& (sigsetjmp(ill_jmp, 1) == 0)) {
|
|
|
|
OPENSSL_vx_probe();
|
|
|
|
} else {
|
|
|
|
OPENSSL_s390xcap_P.stfle[2] &= ~(S390X_CAPBIT(S390X_VX)
|
|
|
|
| S390X_CAPBIT(S390X_VXD)
|
|
|
|
| S390X_CAPBIT(S390X_VXE));
|
|
|
|
}
|
|
|
|
|
|
|
|
sigaction(SIGFPE, &oact, NULL);
|
2015-01-22 11:40:55 +08:00
|
|
|
sigaction(SIGILL, &oact, NULL);
|
|
|
|
sigprocmask(SIG_SETMASK, &oset, NULL);
|
|
|
|
}
|