Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 1 | // 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 Tao | 6a186d3 | 2011-04-20 09:57:05 +1000 | [diff] [blame] | 5 | // Package subtle implements functions that are often useful in cryptographic |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 6 | // code but require careful thought to use correctly. |
| 7 | package subtle |
| 8 | |
David Symonds | 446bfff | 2014-07-22 10:08:23 +1000 | [diff] [blame] | 9 | // ConstantTimeCompare returns 1 iff the two slices, x |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 10 | // and y, have equal contents. The time taken is a function of the length of |
| 11 | // the slices and is independent of the contents. |
| 12 | func ConstantTimeCompare(x, y []byte) int { |
Adam Langley | 384f438 | 2014-02-12 11:58:48 -0500 | [diff] [blame] | 13 | if len(x) != len(y) { |
David Symonds | 446bfff | 2014-07-22 10:08:23 +1000 | [diff] [blame] | 14 | return 0 |
Adam Langley | 384f438 | 2014-02-12 11:58:48 -0500 | [diff] [blame] | 15 | } |
| 16 | |
Robert Griesemer | 5a1d332 | 2009-12-15 15:33:31 -0800 | [diff] [blame] | 17 | var v byte |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 18 | |
| 19 | for i := 0; i < len(x); i++ { |
Robert Griesemer | baba292 | 2009-11-09 21:13:17 -0800 | [diff] [blame] | 20 | v |= x[i] ^ y[i] |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 21 | } |
| 22 | |
Robert Griesemer | 5a1d332 | 2009-12-15 15:33:31 -0800 | [diff] [blame] | 23 | return ConstantTimeByteEq(v, 0) |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 24 | } |
| 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 Griesemer | 5a1d332 | 2009-12-15 15:33:31 -0800 | [diff] [blame] | 28 | func ConstantTimeSelect(v, x, y int) int { return ^(v-1)&x | (v-1)&y } |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 29 | |
Russ Cox | 933d185 | 2009-11-14 10:28:53 -0800 | [diff] [blame] | 30 | // ConstantTimeByteEq returns 1 if x == y and 0 otherwise. |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 31 | func ConstantTimeByteEq(x, y uint8) int { |
Robert Griesemer | 5a1d332 | 2009-12-15 15:33:31 -0800 | [diff] [blame] | 32 | z := ^(x ^ y) |
| 33 | z &= z >> 4 |
| 34 | z &= z >> 2 |
| 35 | z &= z >> 1 |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 36 | |
Robert Griesemer | 5a1d332 | 2009-12-15 15:33:31 -0800 | [diff] [blame] | 37 | return int(z) |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 38 | } |
| 39 | |
| 40 | // ConstantTimeEq returns 1 if x == y and 0 otherwise. |
| 41 | func ConstantTimeEq(x, y int32) int { |
Robert Griesemer | 5a1d332 | 2009-12-15 15:33:31 -0800 | [diff] [blame] | 42 | z := ^(x ^ y) |
| 43 | z &= z >> 16 |
| 44 | z &= z >> 8 |
| 45 | z &= z >> 4 |
| 46 | z &= z >> 2 |
| 47 | z &= z >> 1 |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 48 | |
Robert Griesemer | 5a1d332 | 2009-12-15 15:33:31 -0800 | [diff] [blame] | 49 | return int(z & 1) |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 50 | } |
| 51 | |
Adam Langley | 372f399 | 2014-07-02 15:28:57 -0700 | [diff] [blame] | 52 | // 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 Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 55 | func ConstantTimeCopy(v int, x, y []byte) { |
Adam Langley | 372f399 | 2014-07-02 15:28:57 -0700 | [diff] [blame] | 56 | if len(x) != len(y) { |
| 57 | panic("subtle: slices have different lengths") |
| 58 | } |
| 59 | |
Robert Griesemer | 5a1d332 | 2009-12-15 15:33:31 -0800 | [diff] [blame] | 60 | xmask := byte(v - 1) |
| 61 | ymask := byte(^(v - 1)) |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 62 | for i := 0; i < len(x); i++ { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 63 | x[i] = x[i]&xmask | y[i]&ymask |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 64 | } |
Adam Langley | ad67a86 | 2009-11-02 11:12:07 -0800 | [diff] [blame] | 65 | } |
Adam Langley | e85e678 | 2013-05-15 10:27:34 -0400 | [diff] [blame] | 66 | |
| 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. |
| 69 | func ConstantTimeLessOrEq(x, y int) int { |
| 70 | x32 := int32(x) |
| 71 | y32 := int32(y) |
| 72 | return int(((x32 - y32 - 1) >> 31) & 1) |
| 73 | } |