2000-10-30 00:15:04 +00:00
|
|
|
/*-
|
|
|
|
* Copyright (c) 2000 Semen Ustimenko <semenu@FreeBSD.org>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* $FreeBSD$
|
|
|
|
*/
|
|
|
|
|
2002-06-12 00:33:17 +00:00
|
|
|
#include <sys/param.h>
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2002-06-12 00:33:17 +00:00
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in_systm.h>
|
2002-05-11 10:54:45 +00:00
|
|
|
#include <netinet/in.h>
|
2002-06-12 00:33:17 +00:00
|
|
|
#include <netinet/ip.h>
|
|
|
|
#include <sys/un.h>
|
2002-03-31 01:36:08 +00:00
|
|
|
|
2000-10-30 00:15:04 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2002-05-17 00:44:54 +00:00
|
|
|
#include <string.h>
|
2000-10-30 00:15:04 +00:00
|
|
|
#include <termios.h>
|
|
|
|
#include <openssl/rc4.h>
|
|
|
|
|
|
|
|
#include "defs.h"
|
|
|
|
#include "mbuf.h"
|
|
|
|
#include "log.h"
|
|
|
|
#include "timer.h"
|
|
|
|
#include "fsm.h"
|
|
|
|
#include "lqr.h"
|
|
|
|
#include "hdlc.h"
|
|
|
|
#include "lcp.h"
|
|
|
|
#include "ccp.h"
|
2001-02-04 22:53:11 +00:00
|
|
|
#include "throughput.h"
|
|
|
|
#include "layer.h"
|
|
|
|
#include "link.h"
|
2000-10-30 00:15:04 +00:00
|
|
|
#include "chap_ms.h"
|
2001-02-04 22:53:11 +00:00
|
|
|
#include "proto.h"
|
2000-10-30 00:15:04 +00:00
|
|
|
#include "mppe.h"
|
2001-09-13 10:03:20 +00:00
|
|
|
#include "ua.h"
|
2002-06-12 00:33:17 +00:00
|
|
|
#include "descriptor.h"
|
|
|
|
#ifndef NORADIUS
|
|
|
|
#include "radius.h"
|
|
|
|
#endif
|
|
|
|
#include "ncpaddr.h"
|
|
|
|
#include "iplist.h"
|
|
|
|
#include "slcompress.h"
|
|
|
|
#include "ipcp.h"
|
|
|
|
#include "ipv6cp.h"
|
|
|
|
#include "filter.h"
|
|
|
|
#include "mp.h"
|
|
|
|
#include "ncp.h"
|
|
|
|
#include "bundle.h"
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Documentation:
|
|
|
|
*
|
|
|
|
* draft-ietf-pppext-mppe-04.txt
|
|
|
|
* draft-ietf-pppext-mppe-keys-02.txt
|
|
|
|
*/
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
#define MPPE_OPT_STATELESS 0x1000000
|
|
|
|
#define MPPE_OPT_COMPRESSED 0x01
|
|
|
|
#define MPPE_OPT_40BIT 0x20
|
|
|
|
#define MPPE_OPT_56BIT 0x80
|
|
|
|
#define MPPE_OPT_128BIT 0x40
|
|
|
|
#define MPPE_OPT_BITMASK 0xe0
|
|
|
|
#define MPPE_OPT_MASK (MPPE_OPT_STATELESS | MPPE_OPT_BITMASK)
|
|
|
|
|
2001-07-07 03:06:20 +00:00
|
|
|
#define MPPE_FLUSHED 0x8000
|
|
|
|
#define MPPE_ENCRYPTED 0x1000
|
|
|
|
#define MPPE_HEADER_BITMASK 0xf000
|
|
|
|
#define MPPE_HEADER_FLAG 0x00ff
|
|
|
|
#define MPPE_HEADER_FLAGMASK 0x00ff
|
|
|
|
#define MPPE_HEADER_FLAGSHIFT 8
|
|
|
|
#define MPPE_HEADER_STATEFUL_KEYCHANGES 16
|
2001-06-18 15:00:22 +00:00
|
|
|
|
2000-10-30 00:15:04 +00:00
|
|
|
struct mppe_state {
|
2001-06-18 15:00:22 +00:00
|
|
|
unsigned stateless : 1;
|
|
|
|
unsigned flushnext : 1;
|
|
|
|
unsigned flushrequired : 1;
|
|
|
|
int cohnum;
|
2004-09-05 01:46:52 +00:00
|
|
|
unsigned keylen; /* 8 or 16 bytes */
|
2001-06-18 15:00:22 +00:00
|
|
|
int keybits; /* 40, 56 or 128 bits */
|
|
|
|
char sesskey[MPPE_KEY_LEN];
|
|
|
|
char mastkey[MPPE_KEY_LEN];
|
|
|
|
RC4_KEY rc4key;
|
2000-10-30 00:15:04 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
int MPPE_MasterKeyValid = 0;
|
2000-11-07 23:19:11 +00:00
|
|
|
int MPPE_IsServer = 0;
|
2000-10-30 00:15:04 +00:00
|
|
|
char MPPE_MasterKey[MPPE_KEY_LEN];
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
/*
|
|
|
|
* The peer has missed a packet. Mark the next output frame to be FLUSHED
|
|
|
|
*/
|
|
|
|
static int
|
2000-10-30 00:15:04 +00:00
|
|
|
MPPEResetOutput(void *v)
|
|
|
|
{
|
2001-06-18 15:00:22 +00:00
|
|
|
struct mppe_state *mop = (struct mppe_state *)v;
|
|
|
|
|
|
|
|
if (mop->stateless)
|
|
|
|
log_Printf(LogCCP, "MPPE: Unexpected output channel reset\n");
|
|
|
|
else {
|
|
|
|
log_Printf(LogCCP, "MPPE: Output channel reset\n");
|
|
|
|
mop->flushnext = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0; /* Ask FSM not to ACK */
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
2000-10-30 00:15:29 +00:00
|
|
|
static void
|
|
|
|
MPPEReduceSessionKey(struct mppe_state *mp)
|
|
|
|
{
|
2000-10-30 00:15:04 +00:00
|
|
|
switch(mp->keybits) {
|
|
|
|
case 40:
|
|
|
|
mp->sesskey[2] = 0x9e;
|
|
|
|
mp->sesskey[1] = 0x26;
|
|
|
|
case 56:
|
|
|
|
mp->sesskey[0] = 0xd1;
|
|
|
|
case 128:
|
2002-05-13 20:25:47 +00:00
|
|
|
break;
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-10-30 00:15:29 +00:00
|
|
|
static void
|
|
|
|
MPPEKeyChange(struct mppe_state *mp)
|
|
|
|
{
|
2000-10-30 00:15:04 +00:00
|
|
|
char InterimKey[MPPE_KEY_LEN];
|
|
|
|
RC4_KEY RC4Key;
|
|
|
|
|
|
|
|
GetNewKeyFromSHA(mp->mastkey, mp->sesskey, mp->keylen, InterimKey);
|
|
|
|
RC4_set_key(&RC4Key, mp->keylen, InterimKey);
|
|
|
|
RC4(&RC4Key, mp->keylen, InterimKey, mp->sesskey);
|
|
|
|
|
|
|
|
MPPEReduceSessionKey(mp);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct mbuf *
|
2004-09-05 01:46:52 +00:00
|
|
|
MPPEOutput(void *v, struct ccp *ccp, struct link *l __unused, int pri __unused,
|
|
|
|
u_short *proto, struct mbuf *mp)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
|
|
|
struct mppe_state *mop = (struct mppe_state *)v;
|
|
|
|
struct mbuf *mo;
|
2001-06-18 15:00:22 +00:00
|
|
|
u_short nproto, prefix;
|
|
|
|
int dictinit, ilen, len;
|
2000-10-30 00:15:04 +00:00
|
|
|
char *rp;
|
|
|
|
|
|
|
|
ilen = m_length(mp);
|
2001-06-18 15:00:22 +00:00
|
|
|
dictinit = 0;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
log_Printf(LogDEBUG, "MPPE: Output: Proto %02x (%d bytes)\n", *proto, ilen);
|
|
|
|
if (*proto < 0x21 && *proto > 0xFA) {
|
|
|
|
log_Printf(LogDEBUG, "MPPE: Output: Not encrypting\n");
|
2001-06-18 15:00:22 +00:00
|
|
|
ccp->compout += ilen;
|
|
|
|
ccp->uncompout += ilen;
|
2000-10-30 00:15:04 +00:00
|
|
|
return mp;
|
|
|
|
}
|
|
|
|
|
|
|
|
log_DumpBp(LogDEBUG, "MPPE: Output: Encrypt packet:", mp);
|
|
|
|
|
|
|
|
/* Get mbuf for prefixes */
|
|
|
|
mo = m_get(4, MB_CCPOUT);
|
|
|
|
mo->m_next = mp;
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
rp = MBUF_CTOP(mo);
|
|
|
|
prefix = MPPE_ENCRYPTED | mop->cohnum;
|
|
|
|
|
|
|
|
if (mop->stateless ||
|
|
|
|
(mop->cohnum & MPPE_HEADER_FLAGMASK) == MPPE_HEADER_FLAG) {
|
|
|
|
/* Change our key */
|
|
|
|
log_Printf(LogDEBUG, "MPPEOutput: Key changed [%d]\n", mop->cohnum);
|
|
|
|
MPPEKeyChange(mop);
|
|
|
|
dictinit = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mop->stateless || mop->flushnext) {
|
|
|
|
prefix |= MPPE_FLUSHED;
|
|
|
|
dictinit = 1;
|
|
|
|
mop->flushnext = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dictinit) {
|
|
|
|
/* Initialise our dictionary */
|
|
|
|
log_Printf(LogDEBUG, "MPPEOutput: Dictionary initialised [%d]\n",
|
|
|
|
mop->cohnum);
|
|
|
|
RC4_set_key(&mop->rc4key, mop->keylen, mop->sesskey);
|
|
|
|
}
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
/* Set MPPE packet prefix */
|
2001-09-13 10:03:20 +00:00
|
|
|
ua_htons(&prefix, rp);
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
/* Save encrypted protocol number */
|
|
|
|
nproto = htons(*proto);
|
|
|
|
RC4(&mop->rc4key, 2, (char *)&nproto, rp + 2);
|
|
|
|
|
|
|
|
/* Encrypt main packet */
|
|
|
|
rp = MBUF_CTOP(mp);
|
|
|
|
RC4(&mop->rc4key, ilen, rp, rp);
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
mop->cohnum++;
|
|
|
|
mop->cohnum &= ~MPPE_HEADER_BITMASK;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
/* Set the protocol number */
|
2000-10-30 00:15:04 +00:00
|
|
|
*proto = ccp_Proto(ccp);
|
2001-06-18 15:00:22 +00:00
|
|
|
len = m_length(mo);
|
|
|
|
ccp->uncompout += ilen;
|
|
|
|
ccp->compout += len;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2000-10-30 00:15:29 +00:00
|
|
|
log_Printf(LogDEBUG, "MPPE: Output: Encrypted: Proto %02x (%d bytes)\n",
|
2001-06-18 15:00:22 +00:00
|
|
|
*proto, len);
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
return mo;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2004-09-05 01:46:52 +00:00
|
|
|
MPPEResetInput(void *v __unused)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
2001-06-18 15:00:22 +00:00
|
|
|
log_Printf(LogCCP, "MPPE: Unexpected input channel ack\n");
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct mbuf *
|
|
|
|
MPPEInput(void *v, struct ccp *ccp, u_short *proto, struct mbuf *mp)
|
|
|
|
{
|
|
|
|
struct mppe_state *mip = (struct mppe_state *)v;
|
|
|
|
u_short prefix;
|
|
|
|
char *rp;
|
2001-06-18 15:00:22 +00:00
|
|
|
int dictinit, flushed, ilen, len, n;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
ilen = m_length(mp);
|
2001-06-18 15:00:22 +00:00
|
|
|
dictinit = 0;
|
|
|
|
ccp->compin += ilen;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
log_Printf(LogDEBUG, "MPPE: Input: Proto %02x (%d bytes)\n", *proto, ilen);
|
|
|
|
log_DumpBp(LogDEBUG, "MPPE: Input: Packet:", mp);
|
|
|
|
|
|
|
|
mp = mbuf_Read(mp, &prefix, 2);
|
|
|
|
prefix = ntohs(prefix);
|
2001-06-18 15:00:22 +00:00
|
|
|
flushed = prefix & MPPE_FLUSHED;
|
|
|
|
prefix &= ~flushed;
|
|
|
|
if ((prefix & MPPE_HEADER_BITMASK) != MPPE_ENCRYPTED) {
|
|
|
|
log_Printf(LogERROR, "MPPE: Input: Invalid packet (flags = 0x%x)\n",
|
|
|
|
(prefix & MPPE_HEADER_BITMASK) | flushed);
|
2000-10-30 00:15:04 +00:00
|
|
|
m_freem(mp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
prefix &= ~MPPE_HEADER_BITMASK;
|
|
|
|
|
|
|
|
if (!flushed && mip->stateless) {
|
|
|
|
log_Printf(LogCCP, "MPPEInput: Packet without MPPE_FLUSHED set"
|
|
|
|
" in stateless mode\n");
|
|
|
|
flushed = MPPE_FLUSHED;
|
|
|
|
/* Should we really continue ? */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mip->stateless) {
|
|
|
|
/* Change our key for each missed packet in stateless mode */
|
|
|
|
while (prefix != mip->cohnum) {
|
|
|
|
log_Printf(LogDEBUG, "MPPEInput: Key changed [%u]\n", prefix);
|
|
|
|
MPPEKeyChange(mip);
|
|
|
|
/*
|
|
|
|
* mip->cohnum contains what we received last time in stateless
|
|
|
|
* mode.
|
|
|
|
*/
|
|
|
|
mip->cohnum++;
|
|
|
|
mip->cohnum &= ~MPPE_HEADER_BITMASK;
|
|
|
|
}
|
|
|
|
dictinit = 1;
|
|
|
|
} else {
|
|
|
|
if (flushed) {
|
|
|
|
/*
|
|
|
|
* We can always process a flushed packet.
|
|
|
|
* Catch up on any outstanding key changes.
|
|
|
|
*/
|
|
|
|
n = (prefix >> MPPE_HEADER_FLAGSHIFT) -
|
|
|
|
(mip->cohnum >> MPPE_HEADER_FLAGSHIFT);
|
2001-07-07 03:06:20 +00:00
|
|
|
if (n < 0)
|
|
|
|
n += MPPE_HEADER_STATEFUL_KEYCHANGES;
|
2001-06-18 15:00:22 +00:00
|
|
|
while (n--) {
|
|
|
|
log_Printf(LogDEBUG, "MPPEInput: Key changed during catchup [%u]\n",
|
|
|
|
prefix);
|
|
|
|
MPPEKeyChange(mip);
|
|
|
|
}
|
|
|
|
mip->flushrequired = 0;
|
|
|
|
mip->cohnum = prefix;
|
|
|
|
dictinit = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mip->flushrequired) {
|
|
|
|
/*
|
|
|
|
* Perhaps we should be lenient if
|
|
|
|
* (prefix & MPPE_HEADER_FLAGMASK) == MPPE_HEADER_FLAG
|
|
|
|
* The spec says that we shouldn't be though....
|
|
|
|
*/
|
|
|
|
log_Printf(LogDEBUG, "MPPE: Not flushed - discarded\n");
|
2001-08-27 10:42:21 +00:00
|
|
|
fsm_Output(&ccp->fsm, CODE_RESETREQ, ccp->fsm.reqid++, NULL, 0,
|
|
|
|
MB_CCPOUT);
|
2001-06-18 15:00:22 +00:00
|
|
|
m_freem(mp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prefix != mip->cohnum) {
|
|
|
|
/*
|
|
|
|
* We're in stateful mode and didn't receive the expected
|
|
|
|
* packet. Send a reset request, but don't tell the CCP layer
|
|
|
|
* about it as we don't expect to receive a Reset ACK !
|
|
|
|
* Guess what... M$ invented this !
|
|
|
|
*/
|
|
|
|
log_Printf(LogCCP, "MPPE: Input: Got seq %u, not %u\n",
|
|
|
|
prefix, mip->cohnum);
|
|
|
|
fsm_Output(&ccp->fsm, CODE_RESETREQ, ccp->fsm.reqid++, NULL, 0,
|
|
|
|
MB_CCPOUT);
|
|
|
|
mip->flushrequired = 1;
|
|
|
|
m_freem(mp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((prefix & MPPE_HEADER_FLAGMASK) == MPPE_HEADER_FLAG) {
|
|
|
|
log_Printf(LogDEBUG, "MPPEInput: Key changed [%u]\n", prefix);
|
|
|
|
MPPEKeyChange(mip);
|
|
|
|
dictinit = 1;
|
|
|
|
} else if (flushed)
|
|
|
|
dictinit = 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* mip->cohnum contains what we expect to receive next time in stateful
|
|
|
|
* mode.
|
|
|
|
*/
|
|
|
|
mip->cohnum++;
|
|
|
|
mip->cohnum &= ~MPPE_HEADER_BITMASK;
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
if (dictinit) {
|
|
|
|
log_Printf(LogDEBUG, "MPPEInput: Dictionary initialised [%u]\n", prefix);
|
|
|
|
RC4_set_key(&mip->rc4key, mip->keylen, mip->sesskey);
|
|
|
|
}
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
mp = mbuf_Read(mp, proto, 2);
|
|
|
|
RC4(&mip->rc4key, 2, (char *)proto, (char *)proto);
|
|
|
|
*proto = ntohs(*proto);
|
|
|
|
|
|
|
|
rp = MBUF_CTOP(mp);
|
2001-06-18 15:00:22 +00:00
|
|
|
len = m_length(mp);
|
|
|
|
RC4(&mip->rc4key, len, rp, rp);
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
log_Printf(LogDEBUG, "MPPEInput: Decrypted: Proto %02x (%d bytes)\n",
|
|
|
|
*proto, len);
|
|
|
|
log_DumpBp(LogDEBUG, "MPPEInput: Decrypted: Packet:", mp);
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
ccp->uncompin += len;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
return mp;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2004-09-05 01:46:52 +00:00
|
|
|
MPPEDictSetup(void *v __unused, struct ccp *ccp __unused,
|
|
|
|
u_short proto __unused, struct mbuf *mp __unused)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
2004-09-05 01:46:52 +00:00
|
|
|
/* Nothing to see here */
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const char *
|
2002-04-16 23:57:09 +00:00
|
|
|
MPPEDispOpts(struct fsm_opt *o)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
2001-06-18 15:00:22 +00:00
|
|
|
static char buf[70];
|
2001-09-13 10:03:20 +00:00
|
|
|
u_int32_t val;
|
2001-06-18 15:00:22 +00:00
|
|
|
char ch;
|
2002-03-13 10:21:19 +00:00
|
|
|
int len, n;
|
2001-06-18 15:00:22 +00:00
|
|
|
|
2001-09-13 10:03:20 +00:00
|
|
|
ua_ntohl(o->data, &val);
|
2002-03-13 10:21:19 +00:00
|
|
|
len = 0;
|
|
|
|
if ((n = snprintf(buf, sizeof buf, "value 0x%08x ", (unsigned)val)) > 0)
|
|
|
|
len += n;
|
2001-06-18 15:00:22 +00:00
|
|
|
if (!(val & MPPE_OPT_BITMASK)) {
|
2002-03-13 10:21:19 +00:00
|
|
|
if ((n = snprintf(buf + len, sizeof buf - len, "(0")) > 0)
|
|
|
|
len += n;
|
2001-06-18 15:00:22 +00:00
|
|
|
} else {
|
|
|
|
ch = '(';
|
|
|
|
if (val & MPPE_OPT_128BIT) {
|
2002-03-13 10:21:19 +00:00
|
|
|
if ((n = snprintf(buf + len, sizeof buf - len, "%c128", ch)) > 0)
|
|
|
|
len += n;
|
2001-06-18 15:00:22 +00:00
|
|
|
ch = '/';
|
|
|
|
}
|
|
|
|
if (val & MPPE_OPT_56BIT) {
|
2002-03-13 10:21:19 +00:00
|
|
|
if ((n = snprintf(buf + len, sizeof buf - len, "%c56", ch)) > 0)
|
|
|
|
len += n;
|
2001-06-18 15:00:22 +00:00
|
|
|
ch = '/';
|
|
|
|
}
|
|
|
|
if (val & MPPE_OPT_40BIT) {
|
2002-03-13 10:21:19 +00:00
|
|
|
if ((n = snprintf(buf + len, sizeof buf - len, "%c40", ch)) > 0)
|
|
|
|
len += n;
|
2001-06-18 15:00:22 +00:00
|
|
|
ch = '/';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-03-13 10:21:19 +00:00
|
|
|
if ((n = snprintf(buf + len, sizeof buf - len, " bits, state%s",
|
|
|
|
(val & MPPE_OPT_STATELESS) ? "less" : "ful")) > 0)
|
|
|
|
len += n;
|
2001-06-18 15:00:22 +00:00
|
|
|
|
|
|
|
if (val & MPPE_OPT_COMPRESSED) {
|
2002-03-13 10:21:19 +00:00
|
|
|
if ((n = snprintf(buf + len, sizeof buf - len, ", compressed")) > 0)
|
|
|
|
len += n;
|
2001-06-18 15:00:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
snprintf(buf + len, sizeof buf - len, ")");
|
|
|
|
|
2000-10-30 00:15:04 +00:00
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2001-02-04 22:53:11 +00:00
|
|
|
static int
|
|
|
|
MPPEUsable(struct fsm *fp)
|
|
|
|
{
|
|
|
|
int ok;
|
2002-06-28 08:46:21 +00:00
|
|
|
#ifndef NORADIUS
|
|
|
|
struct radius *r = &fp->bundle->radius;
|
2001-02-04 22:53:11 +00:00
|
|
|
|
2002-06-28 08:46:21 +00:00
|
|
|
/*
|
|
|
|
* If the radius server gave us RAD_MICROSOFT_MS_MPPE_ENCRYPTION_TYPES,
|
|
|
|
* use that instead of our configuration value.
|
|
|
|
*/
|
|
|
|
if (*r->cfg.file) {
|
|
|
|
ok = r->mppe.sendkeylen && r->mppe.recvkeylen;
|
|
|
|
if (!ok)
|
|
|
|
log_Printf(LogCCP, "MPPE: Not permitted by RADIUS server\n");
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
struct lcp *lcp = &fp->link->lcp;
|
|
|
|
ok = (lcp->want_auth == PROTO_CHAP && lcp->want_authtype == 0x81) ||
|
|
|
|
(lcp->his_auth == PROTO_CHAP && lcp->his_authtype == 0x81);
|
|
|
|
if (!ok)
|
|
|
|
log_Printf(LogCCP, "MPPE: Not usable without CHAP81\n");
|
|
|
|
}
|
2001-02-04 22:53:11 +00:00
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
static int
|
|
|
|
MPPERequired(struct fsm *fp)
|
|
|
|
{
|
2002-06-12 00:33:17 +00:00
|
|
|
#ifndef NORADIUS
|
|
|
|
/*
|
|
|
|
* If the radius server gave us RAD_MICROSOFT_MS_MPPE_ENCRYPTION_POLICY,
|
|
|
|
* use that instead of our configuration value.
|
|
|
|
*/
|
|
|
|
if (*fp->bundle->radius.cfg.file && fp->bundle->radius.mppe.policy)
|
|
|
|
return fp->bundle->radius.mppe.policy == MPPE_POLICY_REQUIRED ? 1 : 0;
|
|
|
|
#endif
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
return fp->link->ccp.cfg.mppe.required;
|
|
|
|
}
|
|
|
|
|
|
|
|
static u_int32_t
|
2002-06-12 00:33:17 +00:00
|
|
|
MPPE_ConfigVal(struct bundle *bundle, const struct ccp_config *cfg)
|
2001-06-18 15:00:22 +00:00
|
|
|
{
|
|
|
|
u_int32_t val;
|
|
|
|
|
|
|
|
val = cfg->mppe.state == MPPE_STATELESS ? MPPE_OPT_STATELESS : 0;
|
2002-06-12 00:33:17 +00:00
|
|
|
#ifndef NORADIUS
|
|
|
|
/*
|
|
|
|
* If the radius server gave us RAD_MICROSOFT_MS_MPPE_ENCRYPTION_TYPES,
|
|
|
|
* use that instead of our configuration value.
|
|
|
|
*/
|
|
|
|
if (*bundle->radius.cfg.file && bundle->radius.mppe.types) {
|
|
|
|
if (bundle->radius.mppe.types & MPPE_TYPE_40BIT)
|
|
|
|
val |= MPPE_OPT_40BIT;
|
|
|
|
if (bundle->radius.mppe.types & MPPE_TYPE_128BIT)
|
|
|
|
val |= MPPE_OPT_128BIT;
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
switch(cfg->mppe.keybits) {
|
|
|
|
case 128:
|
|
|
|
val |= MPPE_OPT_128BIT;
|
|
|
|
break;
|
|
|
|
case 56:
|
|
|
|
val |= MPPE_OPT_56BIT;
|
|
|
|
break;
|
|
|
|
case 40:
|
|
|
|
val |= MPPE_OPT_40BIT;
|
|
|
|
break;
|
|
|
|
case 0:
|
|
|
|
val |= MPPE_OPT_128BIT | MPPE_OPT_56BIT | MPPE_OPT_40BIT;
|
|
|
|
break;
|
|
|
|
}
|
2001-06-18 15:00:22 +00:00
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* What options should we use for our first configure request
|
|
|
|
*/
|
2000-10-30 00:15:04 +00:00
|
|
|
static void
|
2002-06-12 00:33:17 +00:00
|
|
|
MPPEInitOptsOutput(struct bundle *bundle, struct fsm_opt *o,
|
|
|
|
const struct ccp_config *cfg)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
2001-09-13 10:03:20 +00:00
|
|
|
u_int32_t mval;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2002-04-16 23:57:09 +00:00
|
|
|
o->hdr.len = 6;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
if (!MPPE_MasterKeyValid) {
|
2000-12-29 22:25:56 +00:00
|
|
|
log_Printf(LogCCP, "MPPE: MasterKey is invalid,"
|
2001-02-04 01:08:24 +00:00
|
|
|
" MPPE is available only with CHAP81 authentication\n");
|
2004-09-05 12:32:20 +00:00
|
|
|
mval = 0;
|
|
|
|
ua_htonl(&mval, o->data);
|
2000-10-30 00:15:04 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2002-06-12 00:33:17 +00:00
|
|
|
|
|
|
|
mval = MPPE_ConfigVal(bundle, cfg);
|
2001-09-13 10:03:20 +00:00
|
|
|
ua_htonl(&mval, o->data);
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
/*
|
|
|
|
* Our CCP request was NAK'd with the given options
|
|
|
|
*/
|
2000-10-30 00:15:04 +00:00
|
|
|
static int
|
2002-06-12 00:33:17 +00:00
|
|
|
MPPESetOptsOutput(struct bundle *bundle, struct fsm_opt *o,
|
|
|
|
const struct ccp_config *cfg)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
2001-09-13 10:03:20 +00:00
|
|
|
u_int32_t mval, peer;
|
|
|
|
|
|
|
|
ua_ntohl(o->data, &peer);
|
2001-06-18 15:00:22 +00:00
|
|
|
|
|
|
|
if (!MPPE_MasterKeyValid)
|
|
|
|
/* Treat their NAK as a REJ */
|
|
|
|
return MODE_NAK;
|
|
|
|
|
2002-06-12 00:33:17 +00:00
|
|
|
mval = MPPE_ConfigVal(bundle, cfg);
|
2001-06-18 15:00:22 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If we haven't been configured with a specific number of keybits, allow
|
|
|
|
* whatever the peer asks for.
|
|
|
|
*/
|
|
|
|
if (!cfg->mppe.keybits) {
|
|
|
|
mval &= ~MPPE_OPT_BITMASK;
|
|
|
|
mval |= (peer & MPPE_OPT_BITMASK);
|
|
|
|
if (!(mval & MPPE_OPT_BITMASK))
|
|
|
|
mval |= MPPE_OPT_128BIT;
|
|
|
|
}
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
/* Adjust our statelessness */
|
|
|
|
if (cfg->mppe.state == MPPE_ANYSTATE) {
|
|
|
|
mval &= ~MPPE_OPT_STATELESS;
|
|
|
|
mval |= (peer & MPPE_OPT_STATELESS);
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
2001-09-13 10:03:20 +00:00
|
|
|
ua_htonl(&mval, o->data);
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
return MODE_ACK;
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
/*
|
|
|
|
* The peer has requested the given options
|
|
|
|
*/
|
2000-10-30 00:15:04 +00:00
|
|
|
static int
|
2002-06-12 00:33:17 +00:00
|
|
|
MPPESetOptsInput(struct bundle *bundle, struct fsm_opt *o,
|
|
|
|
const struct ccp_config *cfg)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
2001-09-13 10:03:20 +00:00
|
|
|
u_int32_t mval, peer;
|
2001-06-18 15:00:22 +00:00
|
|
|
int res = MODE_ACK;
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-09-13 10:03:20 +00:00
|
|
|
ua_ntohl(o->data, &peer);
|
2000-10-30 00:15:04 +00:00
|
|
|
if (!MPPE_MasterKeyValid) {
|
2001-09-13 10:03:20 +00:00
|
|
|
if (peer != 0) {
|
|
|
|
peer = 0;
|
|
|
|
ua_htonl(&peer, o->data);
|
2000-10-30 00:15:04 +00:00
|
|
|
return MODE_NAK;
|
2001-06-18 15:00:22 +00:00
|
|
|
} else
|
2000-10-30 00:15:04 +00:00
|
|
|
return MODE_ACK;
|
|
|
|
}
|
|
|
|
|
2002-06-12 00:33:17 +00:00
|
|
|
mval = MPPE_ConfigVal(bundle, cfg);
|
2001-06-18 15:00:22 +00:00
|
|
|
|
|
|
|
if (peer & ~MPPE_OPT_MASK)
|
|
|
|
/* He's asking for bits we don't know about */
|
|
|
|
res = MODE_NAK;
|
|
|
|
|
|
|
|
if (peer & MPPE_OPT_STATELESS) {
|
|
|
|
if (cfg->mppe.state == MPPE_STATEFUL)
|
|
|
|
/* Peer can't have stateless */
|
|
|
|
res = MODE_NAK;
|
|
|
|
else
|
|
|
|
/* Peer wants stateless, that's ok */
|
|
|
|
mval |= MPPE_OPT_STATELESS;
|
|
|
|
} else {
|
|
|
|
if (cfg->mppe.state == MPPE_STATELESS)
|
|
|
|
/* Peer must have stateless */
|
|
|
|
res = MODE_NAK;
|
|
|
|
else
|
|
|
|
/* Peer doesn't want stateless, that's ok */
|
|
|
|
mval &= ~MPPE_OPT_STATELESS;
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
/* If we've got a configured number of keybits - the peer must use that */
|
|
|
|
if (cfg->mppe.keybits) {
|
2001-09-13 10:03:20 +00:00
|
|
|
ua_htonl(&mval, o->data);
|
2001-06-18 15:00:22 +00:00
|
|
|
return peer == mval ? res : MODE_NAK;
|
|
|
|
}
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
/* If a specific number of bits hasn't been requested, we'll need to NAK */
|
|
|
|
switch (peer & MPPE_OPT_BITMASK) {
|
|
|
|
case MPPE_OPT_128BIT:
|
|
|
|
case MPPE_OPT_56BIT:
|
|
|
|
case MPPE_OPT_40BIT:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
res = MODE_NAK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Suggest the best number of bits */
|
|
|
|
mval &= ~MPPE_OPT_BITMASK;
|
|
|
|
if (peer & MPPE_OPT_128BIT)
|
|
|
|
mval |= MPPE_OPT_128BIT;
|
|
|
|
else if (peer & MPPE_OPT_56BIT)
|
|
|
|
mval |= MPPE_OPT_56BIT;
|
|
|
|
else if (peer & MPPE_OPT_40BIT)
|
|
|
|
mval |= MPPE_OPT_40BIT;
|
|
|
|
else
|
|
|
|
mval |= MPPE_OPT_128BIT;
|
2001-09-13 10:03:20 +00:00
|
|
|
ua_htonl(&mval, o->data);
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct mppe_state *
|
2002-04-16 23:57:09 +00:00
|
|
|
MPPE_InitState(struct fsm_opt *o)
|
2001-06-18 15:00:22 +00:00
|
|
|
{
|
|
|
|
struct mppe_state *mp;
|
|
|
|
u_int32_t val;
|
|
|
|
|
|
|
|
if ((mp = calloc(1, sizeof *mp)) != NULL) {
|
2001-09-13 10:03:20 +00:00
|
|
|
ua_ntohl(o->data, &val);
|
2001-06-18 15:00:22 +00:00
|
|
|
|
|
|
|
switch (val & MPPE_OPT_BITMASK) {
|
|
|
|
case MPPE_OPT_128BIT:
|
|
|
|
mp->keylen = 16;
|
|
|
|
mp->keybits = 128;
|
|
|
|
break;
|
|
|
|
case MPPE_OPT_56BIT:
|
|
|
|
mp->keylen = 8;
|
|
|
|
mp->keybits = 56;
|
|
|
|
break;
|
|
|
|
case MPPE_OPT_40BIT:
|
|
|
|
mp->keylen = 8;
|
|
|
|
mp->keybits = 40;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
log_Printf(LogWARN, "Unexpected MPPE options 0x%08x\n", val);
|
|
|
|
free(mp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
mp->stateless = !!(val & MPPE_OPT_STATELESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
return mp;
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void *
|
2002-06-12 00:33:17 +00:00
|
|
|
MPPEInitInput(struct bundle *bundle, struct fsm_opt *o)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
|
|
|
struct mppe_state *mip;
|
|
|
|
|
|
|
|
if (!MPPE_MasterKeyValid) {
|
2001-02-04 01:08:24 +00:00
|
|
|
log_Printf(LogWARN, "MPPE: Cannot initialise without CHAP81\n");
|
2000-10-30 00:15:04 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
if ((mip = MPPE_InitState(o)) == NULL) {
|
|
|
|
log_Printf(LogWARN, "MPPEInput: Cannot initialise - unexpected options\n");
|
|
|
|
return NULL;
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
log_Printf(LogDEBUG, "MPPE: InitInput: %d-bits\n", mip->keybits);
|
|
|
|
|
2002-06-12 00:33:17 +00:00
|
|
|
#ifndef NORADIUS
|
|
|
|
if (*bundle->radius.cfg.file && bundle->radius.mppe.recvkey) {
|
2002-07-02 00:47:24 +00:00
|
|
|
if (mip->keylen > bundle->radius.mppe.recvkeylen)
|
|
|
|
mip->keylen = bundle->radius.mppe.recvkeylen;
|
2002-06-12 00:33:17 +00:00
|
|
|
if (mip->keylen > sizeof mip->mastkey)
|
|
|
|
mip->keylen = sizeof mip->mastkey;
|
|
|
|
memcpy(mip->mastkey, bundle->radius.mppe.recvkey, mip->keylen);
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
GetAsymetricStartKey(MPPE_MasterKey, mip->mastkey, mip->keylen, 0,
|
|
|
|
MPPE_IsServer);
|
|
|
|
|
2000-10-30 00:15:04 +00:00
|
|
|
GetNewKeyFromSHA(mip->mastkey, mip->mastkey, mip->keylen, mip->sesskey);
|
|
|
|
|
|
|
|
MPPEReduceSessionKey(mip);
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
log_Printf(LogCCP, "MPPE: Input channel initiated\n");
|
|
|
|
|
|
|
|
if (!mip->stateless) {
|
|
|
|
/*
|
|
|
|
* We need to initialise our dictionary here as the first packet we
|
|
|
|
* receive is unlikely to have the FLUSHED bit set.
|
|
|
|
*/
|
|
|
|
log_Printf(LogDEBUG, "MPPEInitInput: Dictionary initialised [%d]\n",
|
|
|
|
mip->cohnum);
|
|
|
|
RC4_set_key(&mip->rc4key, mip->keylen, mip->sesskey);
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* We do the first key change here as the first packet is expected
|
|
|
|
* to have a sequence number of 0 and we'll therefore not expect
|
|
|
|
* to have to change the key at that point.
|
|
|
|
*/
|
|
|
|
log_Printf(LogDEBUG, "MPPEInitInput: Key changed [%d]\n", mip->cohnum);
|
|
|
|
MPPEKeyChange(mip);
|
|
|
|
}
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
return mip;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *
|
2002-06-12 00:33:17 +00:00
|
|
|
MPPEInitOutput(struct bundle *bundle, struct fsm_opt *o)
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
|
|
|
struct mppe_state *mop;
|
|
|
|
|
|
|
|
if (!MPPE_MasterKeyValid) {
|
2001-02-04 01:08:24 +00:00
|
|
|
log_Printf(LogWARN, "MPPE: Cannot initialise without CHAP81\n");
|
2000-10-30 00:15:04 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
if ((mop = MPPE_InitState(o)) == NULL) {
|
|
|
|
log_Printf(LogWARN, "MPPEOutput: Cannot initialise - unexpected options\n");
|
|
|
|
return NULL;
|
2000-10-30 00:15:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
log_Printf(LogDEBUG, "MPPE: InitOutput: %d-bits\n", mop->keybits);
|
|
|
|
|
2002-06-12 00:33:17 +00:00
|
|
|
#ifndef NORADIUS
|
|
|
|
if (*bundle->radius.cfg.file && bundle->radius.mppe.sendkey) {
|
2002-07-02 00:47:24 +00:00
|
|
|
if (mop->keylen > bundle->radius.mppe.sendkeylen)
|
|
|
|
mop->keylen = bundle->radius.mppe.sendkeylen;
|
2002-06-12 00:33:17 +00:00
|
|
|
if (mop->keylen > sizeof mop->mastkey)
|
|
|
|
mop->keylen = sizeof mop->mastkey;
|
|
|
|
memcpy(mop->mastkey, bundle->radius.mppe.sendkey, mop->keylen);
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
GetAsymetricStartKey(MPPE_MasterKey, mop->mastkey, mop->keylen, 1,
|
|
|
|
MPPE_IsServer);
|
|
|
|
|
2000-10-30 00:15:04 +00:00
|
|
|
GetNewKeyFromSHA(mop->mastkey, mop->mastkey, mop->keylen, mop->sesskey);
|
|
|
|
|
|
|
|
MPPEReduceSessionKey(mop);
|
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
log_Printf(LogCCP, "MPPE: Output channel initiated\n");
|
2000-10-30 00:15:04 +00:00
|
|
|
|
2001-06-18 15:00:22 +00:00
|
|
|
if (!mop->stateless) {
|
|
|
|
/*
|
|
|
|
* We need to initialise our dictionary now as the first packet we
|
|
|
|
* send won't have the FLUSHED bit set.
|
|
|
|
*/
|
|
|
|
log_Printf(LogDEBUG, "MPPEInitOutput: Dictionary initialised [%d]\n",
|
|
|
|
mop->cohnum);
|
|
|
|
RC4_set_key(&mop->rc4key, mop->keylen, mop->sesskey);
|
|
|
|
}
|
2000-10-30 00:15:04 +00:00
|
|
|
|
|
|
|
return mop;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
MPPETermInput(void *v)
|
|
|
|
{
|
|
|
|
free(v);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
MPPETermOutput(void *v)
|
|
|
|
{
|
|
|
|
free(v);
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct ccp_algorithm MPPEAlgorithm = {
|
|
|
|
TY_MPPE,
|
|
|
|
CCP_NEG_MPPE,
|
|
|
|
MPPEDispOpts,
|
2001-02-04 22:53:11 +00:00
|
|
|
MPPEUsable,
|
2001-06-18 15:00:22 +00:00
|
|
|
MPPERequired,
|
2000-10-30 00:15:04 +00:00
|
|
|
{
|
|
|
|
MPPESetOptsInput,
|
|
|
|
MPPEInitInput,
|
|
|
|
MPPETermInput,
|
|
|
|
MPPEResetInput,
|
|
|
|
MPPEInput,
|
|
|
|
MPPEDictSetup
|
|
|
|
},
|
|
|
|
{
|
2001-07-03 22:20:19 +00:00
|
|
|
2,
|
2000-10-30 00:15:04 +00:00
|
|
|
MPPEInitOptsOutput,
|
|
|
|
MPPESetOptsOutput,
|
|
|
|
MPPEInitOutput,
|
|
|
|
MPPETermOutput,
|
|
|
|
MPPEResetOutput,
|
|
|
|
MPPEOutput
|
|
|
|
},
|
|
|
|
};
|