blob: 6f80e7c58dc3c8d551b05c8a171b3da9fed6b006 [file] [log] [blame]
Adam Langleyad67a862009-11-02 11:12:07 -08001// Copyright 2009 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
Nigel Tao6a186d32011-04-20 09:57:05 +10005// Package subtle implements functions that are often useful in cryptographic
Adam Langleyad67a862009-11-02 11:12:07 -08006// code but require careful thought to use correctly.
7package subtle
8
David Symonds446bfff2014-07-22 10:08:23 +10009// ConstantTimeCompare returns 1 iff the two slices, x
Adam Langleyad67a862009-11-02 11:12:07 -080010// and y, have equal contents. The time taken is a function of the length of
11// the slices and is independent of the contents.
12func ConstantTimeCompare(x, y []byte) int {
Adam Langley384f4382014-02-12 11:58:48 -050013 if len(x) != len(y) {
David Symonds446bfff2014-07-22 10:08:23 +100014 return 0
Adam Langley384f4382014-02-12 11:58:48 -050015 }
16
Robert Griesemer5a1d3322009-12-15 15:33:31 -080017 var v byte
Adam Langleyad67a862009-11-02 11:12:07 -080018
19 for i := 0; i < len(x); i++ {
Robert Griesemerbaba2922009-11-09 21:13:17 -080020 v |= x[i] ^ y[i]
Adam Langleyad67a862009-11-02 11:12:07 -080021 }
22
Robert Griesemer5a1d3322009-12-15 15:33:31 -080023 return ConstantTimeByteEq(v, 0)
Adam Langleyad67a862009-11-02 11:12:07 -080024}
25
26// ConstantTimeSelect returns x if v is 1 and y if v is 0.
27// Its behavior is undefined if v takes any other value.
Robert Griesemer5a1d3322009-12-15 15:33:31 -080028func ConstantTimeSelect(v, x, y int) int { return ^(v-1)&x | (v-1)&y }
Adam Langleyad67a862009-11-02 11:12:07 -080029
Russ Cox933d1852009-11-14 10:28:53 -080030// ConstantTimeByteEq returns 1 if x == y and 0 otherwise.
Adam Langleyad67a862009-11-02 11:12:07 -080031func ConstantTimeByteEq(x, y uint8) int {
Robert Griesemer5a1d3322009-12-15 15:33:31 -080032 z := ^(x ^ y)
33 z &= z >> 4
34 z &= z >> 2
35 z &= z >> 1
Adam Langleyad67a862009-11-02 11:12:07 -080036
Robert Griesemer5a1d3322009-12-15 15:33:31 -080037 return int(z)
Adam Langleyad67a862009-11-02 11:12:07 -080038}
39
40// ConstantTimeEq returns 1 if x == y and 0 otherwise.
41func ConstantTimeEq(x, y int32) int {
Robert Griesemer5a1d3322009-12-15 15:33:31 -080042 z := ^(x ^ y)
43 z &= z >> 16
44 z &= z >> 8
45 z &= z >> 4
46 z &= z >> 2
47 z &= z >> 1
Adam Langleyad67a862009-11-02 11:12:07 -080048
Robert Griesemer5a1d3322009-12-15 15:33:31 -080049 return int(z & 1)
Adam Langleyad67a862009-11-02 11:12:07 -080050}
51
Adam Langley372f3992014-07-02 15:28:57 -070052// ConstantTimeCopy copies the contents of y into x (a slice of equal length)
53// if v == 1. If v == 0, x is left unchanged. Its behavior is undefined if v
54// takes any other value.
Adam Langleyad67a862009-11-02 11:12:07 -080055func ConstantTimeCopy(v int, x, y []byte) {
Adam Langley372f3992014-07-02 15:28:57 -070056 if len(x) != len(y) {
57 panic("subtle: slices have different lengths")
58 }
59
Robert Griesemer5a1d3322009-12-15 15:33:31 -080060 xmask := byte(v - 1)
61 ymask := byte(^(v - 1))
Adam Langleyad67a862009-11-02 11:12:07 -080062 for i := 0; i < len(x); i++ {
Robert Griesemer40621d52009-11-09 12:07:39 -080063 x[i] = x[i]&xmask | y[i]&ymask
Adam Langleyad67a862009-11-02 11:12:07 -080064 }
Adam Langleyad67a862009-11-02 11:12:07 -080065}
Adam Langleye85e6782013-05-15 10:27:34 -040066
67// ConstantTimeLessOrEq returns 1 if x <= y and 0 otherwise.
68// Its behavior is undefined if x or y are negative or > 2**31 - 1.
69func ConstantTimeLessOrEq(x, y int) int {
70 x32 := int32(x)
71 y32 := int32(y)
72 return int(((x32 - y32 - 1) >> 31) & 1)
73}