cafuncs.C
Upload User: zbbssh
Upload Date: 2007-01-08
Package Size: 196k
Code Size: 7k
Category:

CA program

Development Platform:

C/C++

  1. /*
  2. ------------------------------------------------------------------
  3.   Copyright
  4.   Sun Microsystems, Inc.
  5.   Copyright (C) 1994, 1995, 1996 Sun Microsystems, Inc.  All Rights
  6.   Reserved.
  7.   Permission is hereby granted, free of charge, to any person
  8.   obtaining a copy of this software and associated documentation
  9.   files (the "Software"), to deal in the Software without
  10.   restriction, including without limitation the rights to use,
  11.   copy, modify, merge, publish, distribute, sublicense, and/or sell
  12.   copies of the Software or derivatives of the Software, and to 
  13.   permit persons to whom the Software or its derivatives is furnished 
  14.   to do so, subject to the following conditions:
  15.   The above copyright notice and this permission notice shall be
  16.   included in all copies or substantial portions of the Software.
  17.   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  18.   EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
  19.   OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  20.   NONINFRINGEMENT.  IN NO EVENT SHALL SUN MICROSYSTEMS, INC., BE LIABLE
  21.   FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  22.   OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
  23.   CONNECTION WITH THE SOFTWARE OR DERIVATES OF THIS SOFTWARE OR 
  24.   THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  25.   Except as contained in this notice, the name of Sun Microsystems, Inc.
  26.   shall not be used in advertising or otherwise to promote
  27.   the sale, use or other dealings in this Software or its derivatives 
  28.   without prior written authorization from Sun Microsystems, Inc.
  29. */
  30. #pragma ident "@(#)cafuncs.C 1.14 96/01/29 Sun Microsystems"
  31. #include <sys/types.h>
  32. #include <stdio.h>
  33. #include <stdlib.h>
  34. #include "Time.h"
  35. #include "Bigint.h"
  36. #include "Bstream.h"
  37. #include "asn1_der.h"
  38. #include "ObjId.h"
  39. #include "Name.h"
  40. #include "X509Cert.h"
  41. #include "Sig.h"
  42. #include "ca.h"
  43. #include "utils.h"
  44. int
  45. get_certreq_params(const Bstream& certreq, Name& name, PubKey& pubkey)
  46. {
  47. int retval, seqlen;
  48. PubKey psubject;
  49. AlgId sigalgid;
  50. Name subname;
  51. Bstream sigdata, signature;
  52. Bigint version;
  53. VerifyResult result = INVALID_SIG;
  54. Bstream der_stream = certreq;
  55. SEQUENCE
  56. {
  57. sigdata = der_stream;
  58. SEQUENCE // X509CertReqInfo --- ToBeSigned
  59. {
  60. INTEGER(version);
  61. NAME(subname);
  62. SEQUENCE // Subject Public Key info
  63. {
  64. SEQUENCE // Subject Public Key Alg ID
  65. {
  66. int begin, end, used, anylen;
  67. MARK_LEN(begin);
  68.      OBJECT_IDENTIFIER(psubject.keytype.algid);
  69. MARK_LEN(end);
  70. used = begin - end;
  71. if ((anylen = seqlen - used) > 0) {
  72. Bstream any(anylen, DATAP);
  73. CONSUME(anylen);
  74. psubject.keytype.params = any;
  75. }
  76. }
  77. //  Subject Public Key
  78. BIT_STRING(psubject.key);
  79. }
  80. }
  81. int tobesignedlen = sigdata.getlength() - 
  82. der_stream.getlength();
  83. int trunclen = sigdata.getlength() - tobesignedlen;
  84. sigdata.truncate(trunclen);
  85. SEQUENCE // Alg ID of certreq signing
  86. {
  87. int begin, end, used, anylen;
  88. MARK_LEN(begin);
  89. OBJECT_IDENTIFIER(sigalgid.algid);
  90. MARK_LEN(end);
  91. used = begin - end;
  92. if ((anylen = seqlen - used) > 0) {
  93. Bstream any(anylen, DATAP);
  94. CONSUME(anylen);
  95. sigalgid.params = any;
  96. }
  97. }
  98. //  encrypted hash, i.e SIGNATURE
  99. BIT_STRING(signature);
  100. }
  101. // Verify the X509Certificate request. This is done
  102. // by using the supplied public key to verify
  103. // the signature, since certificate requests are
  104. // self-signed.
  105. // Unfortunately, Diffie-Hellman X.509 certificate requests
  106. // are not self-signed ...
  107. if (psubject.keytype.algid == dhKeyAgreement) {
  108. pubkey = psubject;
  109. name = subname;
  110. result = VALID;
  111. } else {
  112. result = verify_sig(sigdata, signature, psubject, sigalgid);
  113. if (result == VALID) {
  114. pubkey = psubject;
  115. name = subname;
  116. } else {
  117. fprintf(stderr,
  118. "Signature on certificate request invalidn");
  119. }
  120. }
  121. return ((int)result);
  122. }
  123. // Get the serial number for the certificate being issued.
  124. // Increments the value of serialnum stored in NEXT_SERIALNUM_FILE
  125. // to return the present value + 1, for the next certificate
  126. // to be issued.
  127. Bigint
  128. getSerialnum()
  129. {
  130. Bstream serialstr = File_to_Bstr(get_byzantine_dir() + NEXT_SERIALNUM_FILE);
  131. Bigint serialnum(serialstr.getdatap(), serialstr.getlength());
  132. if (serialnum == (short)0)
  133. serialnum = serialnum + 1;
  134. serialstr = Bigint_to_Bstr(serialnum + 1);
  135. serialstr.store(get_byzantine_dir() + NEXT_SERIALNUM_FILE);
  136. return (serialnum);
  137. }
  138. X509Cert
  139. getX509CertbySerialNumber(const Bigint& serialnum)
  140. {
  141. X509Cert cert;
  142. Bstream nullbstr;
  143. String certfilename = (String)CERT_DATABASE_DIR + "/" + 
  144. serialnum.getnumstr();
  145. String certpath = get_byzantine_dir() + certfilename;
  146. Bstream certstr = File_to_Bstr(certpath);
  147. if (certstr == nullbstr) {
  148. fprintf(stderr, "Unable to read certificate file '%s'.n",
  149. (const char *)certpath);
  150. return (cert);
  151. }
  152. int retval = asn1_der_decode_cert(certstr, cert);
  153. if (retval < 0) {
  154. fprintf(stderr, "Unable to get certificate for serial number ");
  155. serialnum.print();
  156. fprintf(stderr, "n");
  157. asn1_perror(retval);
  158. }
  159. return (cert);
  160. }
  161. X509Cert
  162. getCAcert()
  163. {
  164. X509Cert cert;
  165. Bstream nullstr;
  166. String certfile = get_byzantine_dir() + CA_CERT_FILE;
  167. Bstream certstr = File_to_Bstr(certfile);
  168. if (certstr == nullstr) {
  169. fprintf(stderr, "No contents in CA self certificate file %sn",
  170. (const char *)certfile);
  171. return (cert);
  172. }
  173. int retval = asn1_der_decode_cert(certstr, cert);
  174. if (retval < 0) {
  175. fprintf(stderr, "Unable to decode CA certificate in file %sn",
  176. (const char *)certfile);
  177. asn1_perror(retval);
  178. }
  179. return (cert);
  180. }
  181. Boolean
  182. allow_certification(Name& subject, PubKey& pub_subject)
  183. {
  184. // get the current serial number
  185. Bstream serialstr = File_to_Bstr(get_byzantine_dir() + NEXT_SERIALNUM_FILE);
  186. Bigint cur_serialnum(serialstr.getdatap(), serialstr.getlength());
  187. X509Cert cert;
  188. Bstream certstr, nullbstr;
  189. String certdir = get_byzantine_dir() + (String)CERT_DATABASE_DIR + "/";
  190. PubKey pubkey;
  191. int retval;
  192. for (Bigint i = 1; i < cur_serialnum; i = i + 1) {
  193. certstr = File_to_Bstr(certdir + i.getnumstr());
  194. retval = asn1_der_decode_cert(certstr, cert);
  195. if (retval < 0) {
  196. asn1_perror(retval);
  197. fprintf(stderr, "skipping .... n");
  198. continue;
  199. }
  200. // check key collisions
  201. pubkey = cert.getpubkey();
  202. if (pubkey.key == pub_subject.key) {
  203. fprintf(stderr, "Public key collision!!n");
  204. return(BOOL_FALSE);
  205. }
  206. // now check to make sure the subject does not already have
  207. // a key with the same parameters
  208. if (cert.getsubject() == subject) {
  209. // found a certificate for this subject
  210. // now check the params => should be different
  211. if (pubkey.keytype.algid == pub_subject.keytype.algid) {
  212. if ((pubkey.keytype.params ==
  213. pub_subject.keytype.params)
  214. // XXX check for different exponent lengths
  215. // || ()
  216. ) {
  217. fprintf(stderr,
  218. "Subject already has a public value with same DH paramsn");
  219. return(BOOL_FALSE);
  220. }
  221. }
  222. }
  223. }
  224. return (BOOL_TRUE);
  225. }