| /*############################################################################ |
| # Copyright 2017 Intel Corporation |
| # |
| # 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. |
| ############################################################################*/ |
| /// Non-sensitive member context implementation |
| /*! \file */ |
| |
| #include "epid/member/src/validatekey.h" |
| |
| #include <stddef.h> |
| |
| #include "epid/common/math/ecgroup.h" |
| #include "epid/common/math/finitefield.h" |
| #include "epid/common/math/pairing.h" |
| #include "epid/common/src/epid2params.h" |
| #include "epid/common/src/memory.h" |
| #include "epid/common/types.h" // MemberPrecomp |
| #include "epid/member/src/context.h" |
| #include "epid/member/src/privateexp.h" |
| |
| /// Handle Intel(R) EPID Error with Break |
| #define BREAK_ON_EPID_ERROR(ret) \ |
| if (kEpidNoErr != (ret)) { \ |
| break; \ |
| } |
| |
| bool EpidMemberIsKeyValid(MemberCtx* ctx, G1ElemStr const* A_str, |
| FpElemStr const* x_str, G1ElemStr const* h1_str, |
| G2ElemStr const* w_str) { |
| bool key_is_valid = false; |
| EcPoint* t1 = NULL; |
| EcPoint* t2 = NULL; |
| FfElement* t3 = NULL; |
| FfElement* t4 = NULL; |
| EcPoint* A = NULL; |
| EcPoint* h1 = NULL; |
| EcPoint* w = NULL; |
| |
| if (!ctx || !A_str || !x_str || !h1_str || !w_str || !ctx->epid2_params) { |
| return false; |
| } |
| |
| do { |
| EpidStatus sts = kEpidErr; |
| EcGroup* G1 = ctx->epid2_params->G1; |
| EcGroup* G2 = ctx->epid2_params->G2; |
| FiniteField* GT = ctx->epid2_params->GT; |
| EcPoint* g1 = ctx->epid2_params->g1; |
| EcPoint* g2 = ctx->epid2_params->g2; |
| PairingState* ps_ctx = ctx->epid2_params->pairing_state; |
| |
| if (!ctx->is_provisioned && !ctx->is_initially_provisioned) { |
| sts = EpidMemberInitialProvision(ctx); |
| BREAK_ON_EPID_ERROR(sts); |
| } |
| |
| // 2. The member computes t1 = G2.sscmExp(g2, x). |
| sts = NewEcPoint(G2, &t1); |
| BREAK_ON_EPID_ERROR(sts); |
| |
| sts = EcSscmExp(G2, g2, (BigNumStr const*)x_str, t1); |
| BREAK_ON_EPID_ERROR(sts); |
| |
| // 3. The member computes t1 = G2.mul(t1, w). |
| sts = NewEcPoint(G2, &w); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = ReadEcPoint(G2, w_str, sizeof(*w_str), w); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = EcMul(G2, t1, w, t1); |
| BREAK_ON_EPID_ERROR(sts); |
| |
| // 4. The member computes t3 = pairing(A, t1). |
| sts = NewFfElement(GT, &t3); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = NewEcPoint(G1, &A); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = ReadEcPoint(G1, A_str, sizeof(*A_str), A); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = Pairing(ps_ctx, A, t1, t3); |
| BREAK_ON_EPID_ERROR(sts); |
| |
| // 5. The member computes t2 = G1.sscmExp(h1, f). |
| sts = NewEcPoint(G1, &t2); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = NewEcPoint(G1, &h1); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = ReadEcPoint(G1, h1_str, sizeof(*h1_str), h1); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = EpidPrivateExp(ctx, h1, t2); |
| BREAK_ON_EPID_ERROR(sts); |
| |
| // 6. The member computes t2 = G1.mul(t2, g1). |
| sts = EcMul(G1, t2, g1, t2); |
| BREAK_ON_EPID_ERROR(sts); |
| |
| // Step 7. The member computes t4 = pairing(t2, g2). |
| sts = NewFfElement(GT, &t4); |
| BREAK_ON_EPID_ERROR(sts); |
| sts = Pairing(ps_ctx, t2, g2, t4); |
| BREAK_ON_EPID_ERROR(sts); |
| |
| // 8. If GT.isEqual(t3, t4) = false, reports bad private key. |
| sts = FfIsEqual(GT, t3, t4, &key_is_valid); |
| if (kEpidNoErr != sts) { |
| key_is_valid = false; |
| BREAK_ON_EPID_ERROR(sts); |
| } |
| } while (0); |
| |
| DeleteEcPoint(&t1); |
| DeleteEcPoint(&t2); |
| DeleteFfElement(&t3); |
| DeleteFfElement(&t4); |
| DeleteEcPoint(&A); |
| DeleteEcPoint(&h1); |
| DeleteEcPoint(&w); |
| |
| return key_is_valid; |
| } |