52
52
*/
53
53
54
54
public final class ObjectIdentifier implements Serializable {
55
+ /*
56
+ * The maximum encoded OID length, excluding the ASN.1 encoding tag and
57
+ * length.
58
+ *
59
+ * In theory, there is no maximum size for OIDs. However, there are some
60
+ * limitation in practice.
61
+ *
62
+ * RFC 5280 mandates support for OIDs that have arc elements with values
63
+ * that are less than 2^28 (that is, they MUST be between 0 and
64
+ * 268,435,455, inclusive), and implementations MUST be able to handle
65
+ * OIDs with up to 20 elements (inclusive). Per RFC 5280, an encoded
66
+ * OID should be less than 80 bytes for safe interoperability.
67
+ *
68
+ * This class could be used for protocols other than X.509 certificates.
69
+ * To be safe, a relatively large but still reasonable value is chosen
70
+ * as the restriction in JDK.
71
+ */
72
+ private static final int MAXIMUM_OID_SIZE = 4096 ; // 2^12
73
+
55
74
/**
56
75
* We use the DER value (no tag, no length) as the internal format
57
76
* @serial
@@ -118,7 +137,13 @@ private void readObject(ObjectInputStream is)
118
137
if (componentLen > comp .length ) {
119
138
componentLen = comp .length ;
120
139
}
140
+
141
+ // Check the estimated size before it is too later.
142
+ checkOidSize (componentLen );
143
+
121
144
init (comp , componentLen );
145
+ } else {
146
+ checkOidSize (encoding .length );
122
147
}
123
148
}
124
149
@@ -203,6 +228,8 @@ private ObjectIdentifier(String oid) throws IOException {
203
228
}
204
229
start = end + 1 ;
205
230
count ++;
231
+
232
+ checkOidSize (pos );
206
233
} while (end != -1 );
207
234
208
235
checkCount (count );
@@ -250,11 +277,13 @@ public ObjectIdentifier(DerInputStream in) throws IOException {
250
277
);
251
278
252
279
int len = in .getDefiniteLength ();
280
+ checkOidSize (len );
253
281
if (len > in .available ()) {
254
- throw new IOException ("ObjectIdentifier() -- length exceeds" +
282
+ throw new IOException ("ObjectIdentifier length exceeds " +
255
283
"data available. Length: " + len + ", Available: " +
256
284
in .available ());
257
285
}
286
+
258
287
encoding = new byte [len ];
259
288
in .getBytes (encoding );
260
289
check (encoding );
@@ -267,26 +296,32 @@ public ObjectIdentifier(DerInputStream in) throws IOException {
267
296
*/
268
297
ObjectIdentifier (DerInputBuffer buf ) throws IOException {
269
298
DerInputStream in = new DerInputStream (buf );
270
- encoding = new byte [in .available ()];
299
+ int len = in .available ();
300
+ checkOidSize (len );
301
+
302
+ encoding = new byte [len ];
271
303
in .getBytes (encoding );
272
304
check (encoding );
273
305
}
274
306
275
- private void init (int [] components , int length ) {
307
+ private void init (int [] components , int length ) throws IOException {
276
308
int pos = 0 ;
277
- byte [] tmp = new byte [length * 5 + 1 ]; // +1 for empty input
309
+ byte [] tmp = new byte [length * 5 + 1 ]; // +1 for empty input
278
310
279
- if (components [1 ] < Integer .MAX_VALUE - components [0 ]* 40 )
280
- pos += pack7Oid (components [0 ]* 40 + components [1 ], tmp , pos );
281
- else {
311
+ if (components [1 ] < Integer .MAX_VALUE - components [0 ] * 40 ) {
312
+ pos += pack7Oid (components [0 ] * 40 + components [1 ], tmp , pos );
313
+ } else {
282
314
BigInteger big = BigInteger .valueOf (components [1 ]);
283
- big = big .add (BigInteger .valueOf (components [0 ]* 40 ));
315
+ big = big .add (BigInteger .valueOf (components [0 ] * 40 ));
284
316
pos += pack7Oid (big , tmp , pos );
285
317
}
286
318
287
- for (int i = 2 ; i < length ; i ++) {
319
+ for (int i = 2 ; i < length ; i ++) {
288
320
pos += pack7Oid (components [i ], tmp , pos );
321
+
322
+ checkOidSize (pos );
289
323
}
324
+
290
325
encoding = new byte [pos ];
291
326
System .arraycopy (tmp , 0 , encoding , 0 , pos );
292
327
}
@@ -690,4 +725,13 @@ private static void checkOtherComponent(
690
725
"oid component #" + (i +1 ) + " must be non-negative " );
691
726
}
692
727
}
728
+
729
+ private static void checkOidSize (int oidLength ) throws IOException {
730
+ if (oidLength > MAXIMUM_OID_SIZE ) {
731
+ throw new IOException (
732
+ "ObjectIdentifier encoded length exceeds " +
733
+ "the restriction in JDK (OId length(>=): " + oidLength +
734
+ ", Restriction: " + MAXIMUM_OID_SIZE + ")" );
735
+ }
736
+ }
693
737
}
0 commit comments