⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 pgptextfilt.c

📁 著名的加密软件的应用于电子邮件中
💻 C
字号:
/*
* pgpTextFilt.c -- filter text
*
* Copyright (C) 1995-1997 Pretty Good Privacy, Inc. All rights reserved.
*
* Written by: Derek Atkins <warlord@MIT.EDU> and Colin Plumb
*
* $Id: pgpTextFilt.c,v 1.4.2.1 1997/06/07 09:51:09 mhw Exp $
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif

#include <stdio.h>

#include "pgpDebug.h"
#include "pgpTextFilt.h"
#include "pgpMem.h"
#include "pgpPipeline.h"
#include "pgpErr.h"
#include "pgpUsuals.h"

#if (BUFSIZ < 16384) && (MACINTOSH || WIN32)
#define kPGPFiltBufSize 16384
#else
#define kPGPFiltBufSize BUFSIZ
#endif

#define TEXTFILTMAGIC 0x1e81f111

struct Context {
		byte buffer[kPGPFiltBufSize];
		byte *bufptr;
		size_t buflen;
		struct PgpPipeline *tail;
		byte const *map;	/* 256-entry mapping table */
		int crlf;	 /* 0=nothing, 1=LF, 2=CRLF, 3=CR */
		int stripspace;
		size_t spaces;
		int ignorelf;
};

static int
DoFlush (struct Context *context)
	{
		int error = 0;
		size_t retlen;

		/* Try to flush anything that we have buffered */
		while (context->buflen) {
			retlen = context->tail->write (context->tail,
			   context->bufptr,
			   context->buflen,
			   &error);
			context->buflen -= retlen;
			memset (context->bufptr, 0, retlen);
			context->bufptr += retlen;
			if (error)
		 	return error;
		}
		return error;
	}

/*
* Note: it is assumed that \r, \n and ' ' use the identity mapping.
* This will need fixing for EBCDIC.
*/
static size_t
textFilt (struct Context *context, byte const *in, size_t inlen)
	{
		size_t spaces = context->spaces;
		int ignorelf = context->ignorelf;
		byte *out = context->bufptr+context->buflen;
		size_t outlen = context->buffer+sizeof(context->buffer)-out;
		size_t inlen0 = inlen;
		byte c;

		pgpAssert (outlen);

		while (inlen && outlen) {
			c = *in;

			if (ignorelf) {
				ignorelf = 0;
				if (c == '\n') {
					in++;
					inlen--;
					continue;
				}
			}
			if (c == '\n' || c == '\r') {
				if (context->crlf) {
					if (context->crlf == PGP_TEXTFILT_CRLF) {
					 if (outlen < 2)
					  break;
					 outlen--;
					}
					if (context->crlf & PGP_TEXTFILT_CR)
					 *out++ = '\r';
					if (context->crlf & PGP_TEXTFILT_LF)
					 *out++ = '\n';
					outlen--;
					in++;
					inlen--;
					ignorelf = (c == '\r');
					spaces = 0;
					continue;
				}
				spaces = 0;
			}
			if (context->stripspace && c == ' ') {
				spaces++;
				in++;
				inlen--;
				continue;
			}
			if (spaces) {
				int i = min (spaces, outlen);

				memset (out, ' ', i);
				outlen -= i;
				spaces -= i;
				out += i;
				continue;
			}
			*out++ = context->map[c];
			in++;
			outlen--;
			inlen--;
		}

		context->spaces = spaces;
		context->ignorelf = ignorelf;
		context->buflen = out - context->bufptr;
		return inlen0 - inlen;
	}

static int
Flush (struct PgpPipeline *myself)
{
struct Context *context;
int error;

pgpAssert (myself);
pgpAssert (myself->magic == TEXTFILTMAGIC);

context = (struct Context *)myself->priv;
pgpAssert (context);
pgpAssert (context->tail);

error = DoFlush (context);
if (error)
 return error;

return context->tail->flush (context->tail);
}

static size_t
Write (struct PgpPipeline *myself, byte const *buf, size_t size, int *error)
	{
		struct Context *context;
		size_t written, retlen = 0;

		pgpAssert (myself);
		pgpAssert (myself->magic == TEXTFILTMAGIC);
		pgpAssert (error);

context = (struct Context *)myself->priv;
pgpAssert (context);
pgpAssert (context->tail);

do {
 *error = DoFlush (context);
 if (*error)
			 return retlen;

			/*
			* Now that we dont have anything buffered, bring in more
			* data from the passed-in buffer, process it, and buffer
			* that to write out.
			*/
			context->bufptr = context->buffer;
			written = textFilt (context, buf, size);
			/* context->buflen is set by textFilt() */
			buf += written;
			size -= written;
			retlen += written;

		} while (context->buflen > 0);
		/* Continue until we have nothing buffered */

		return retlen;
	}

static int
Annotate (struct PgpPipeline *myself, struct PgpPipeline *origin, int type,
		byte const *string, size_t size)
	{
		struct Context *context;
		int error;

		pgpAssert (myself);
		pgpAssert (myself->magic == TEXTFILTMAGIC);

		context = (struct Context *)myself->priv;
		pgpAssert (context);
		pgpAssert (context->tail);

		error = DoFlush (context);
		if (error)
		 return error;

		return context->tail->annotate (context->tail, origin, type,
		    string, size);
	}

static int
SizeAdvise (struct PgpPipeline *myself, unsigned long bytes)
{
struct Context *context;
int error;

pgpAssert (myself);
pgpAssert (myself->magic == TEXTFILTMAGIC);

context = (struct Context *)myself->priv;
pgpAssert (context);
pgpAssert (context->tail);

error = DoFlush (context);
if (error || (bytes && (context->stripspace || context->crlf)))
 return error;

  return context->tail->sizeAdvise (context->tail, bytes);
}

static void
Teardown (struct PgpPipeline *myself)
	{
		struct Context *context;

		pgpAssert (myself);
		pgpAssert (myself->magic == TEXTFILTMAGIC);

		context = (struct Context *)myself->priv;
		pgpAssert (context);

		if (context->tail)
		 context->tail->teardown (context->tail);

		memset (context, 0, sizeof (*context));
		pgpMemFree (context);
		memset (myself, 0, sizeof (*myself));
		pgpMemFree (myself);
	}

struct PgpPipeline **
pgpTextFiltCreate (struct PgpPipeline **head, byte const *map, int stripspace,
		 int crlf)
	{
		struct PgpPipeline *mod;
		struct Context *context;

		pgpAssert (map);

if (!head)
 return NULL;

context = (struct Context *)pgpMemAlloc (sizeof (*context));
if (!context)
 return NULL;
mod = (struct PgpPipeline *)pgpMemAlloc (sizeof (*mod));
if (!mod){
 pgpMemFree (context);
 return NULL;
}

		mod->magic = TEXTFILTMAGIC;
		mod->write = Write;
		mod->flush = Flush;
		mod->sizeAdvise = SizeAdvise;
		mod->annotate = Annotate;
		mod->teardown = Teardown;
		mod->name = "Text Filter Module";
		mod->priv = context;

		memset (context, 0, sizeof (*context));
		context->bufptr = context->buffer;
		context->map = map;
		context->stripspace = stripspace;
context->crlf = crlf;

context->tail = *head;
*head = mod;
return &context->tail;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -