You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

395 lines
11 KiB

/*
* $Id: versamail.c,v 1.11 2006/10/12 14:21:23 desrod Exp $
*
* versamail.c: Translate VersaMail data formats
*
* Copyright (c) 2004, Nick Piper
*
* This library is free software; you can redistribute it and/or modify it
* under the terms of the GNU Library General Public License as published by
* the Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This library is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library
* General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifdef TIME_WITH_SYS_TIME
# include <sys/time.h>
# include <time.h>
#else
# ifdef HAVE_SYS_TIME_H
# include <sys/time.h>
# else
# include <time.h>
# endif
#endif
#include "pi-macros.h"
#include "pi-versamail.h"
/*
Todo: What is unknown1 and 2?
What is reserved 1 and 2?
What is download ? (It's non zero, when there is more to d/l,
and appears to be an ID, allocated
by VersaMail. Maybe an index into another db?)
How do we do attachments ? They are not kept in-line, but in another db.
Attachments are related to unknown3.
Why is mark = 2, when there is no mark set, and 3 when it is?
*/
void free_VersaMail(struct VersaMail *a)
{
if (a->messageUID)
free(a->messageUID);
if (a->to)
free(a->to);
if (a->from)
free(a->from);
if (a->cc)
free(a->cc);
if (a->bcc)
free(a->bcc);
if (a->subject)
free(a->subject);
if (a->dateString)
free(a->dateString);
if (a->body)
free(a->body);
if (a->replyTo)
free(a->replyTo);
if (a->unknown3)
free(a->unknown3);
}
#define SHUFFLE_BUFFER(shuffle_distance) { \
len -= shuffle_distance; \
buffer += shuffle_distance; \
}
#define GET_BYTE_INTO(attrib) { \
attrib = get_byte(buffer); \
SHUFFLE_BUFFER(1); \
}
#define GET_SHORT_INTO(attrib) { \
attrib = (unsigned int) get_short(buffer); \
SHUFFLE_BUFFER(2); \
}
#define GET_STR_INTO(attrib) { \
if (get_byte(buffer)) { \
attrib = strdup(buffer); \
SHUFFLE_BUFFER(strlen(buffer)); \
} else { \
attrib = NULL; \
}; \
buffer++; \
len--; \
}
#define PUT_STR_FROM(attrib) { \
if (attrib) { \
strcpy(buffer, attrib); \
SHUFFLE_BUFFER(strlen(buffer)); \
} else \
set_byte(buffer, 0); \
buffer++; \
len--; \
}
#define ADD_LENGTH(str, destlen) { \
if (str) { \
destlen += 1+strlen(str); \
} else { \
destlen++; \
}; \
} \
#define CONVERT_TIME_T_PALM_TO_UNIX(t) (t=t-2082844800)
#define CONVERT_TIME_T_UNIX_TO_PALM(t) (t=t+2082844800)
int unpack_VersaMail(struct VersaMail *a, char *buffer, size_t len)
{
time_t date_t;
struct tm *date_tm;
char *start = buffer;
/* 000000 - 000003 */
a->imapuid = (unsigned long) get_long(buffer);
SHUFFLE_BUFFER(4);
/* This is different to the other databases, because Palm didn't
write VersaMail :-) */
/* 000004 - 000007 */
date_t = (time_t) get_long(buffer);
CONVERT_TIME_T_PALM_TO_UNIX(date_t);
date_tm = localtime(&date_t);
memcpy(&(a->date), date_tm, sizeof(struct tm));
SHUFFLE_BUFFER(4);
/* 000008 - 000009 */
GET_SHORT_INTO(a->category);
/* 00000A - 00000B */
GET_SHORT_INTO(a->accountNo);
/* 00000C - 00000D */
GET_SHORT_INTO(a->unknown1);
/* 00000E - 00000E */
GET_BYTE_INTO(a->download);
/* 00000F - 00000F */
GET_BYTE_INTO(a->mark);
/* of the above, bit 0 is mark, apparently bit 4 is header-only flag,
and it looks like bit 1 is always set, which gives a normal
value of 2 */
/* 000010 - 000011 */
GET_SHORT_INTO(a->unknown2);
/* 000012 - 000013 */
a->reserved1 = (get_byte(buffer));
a->reserved2 = (get_byte(buffer + 1) >> 1);
a->read = (get_byte(buffer + 1) && 1);
SHUFFLE_BUFFER(2);
/* This is the size, as provided by the imap server:
* 1 FETCH (UID 12779 BODY[] {3377} .... )
Size==3377
I guess VersaMail uses it to determine how much
more there might be to download. I don't think it uses it
to know there IS more to download though, because
the palm doesn't track most of the headers, so it'll never
be able to calculate if there is more or not, exactly.
*/
/* 000014 - 000017 */
a->msgSize = get_long(buffer);
SHUFFLE_BUFFER(4);
GET_STR_INTO(a->messageUID);
GET_STR_INTO(a->to);
GET_STR_INTO(a->from);
GET_STR_INTO(a->cc);
GET_STR_INTO(a->bcc);
GET_STR_INTO(a->subject);
GET_STR_INTO(a->dateString);
GET_STR_INTO(a->body);
GET_STR_INTO(a->replyTo);
a->unknown3length = 0;
a->unknown3 = NULL;
a->attachmentCount = 0;
if (len > 0) {
a->unknown3 = (void *) malloc(len);
/*
printf("Msg has extra %d bytes. That's 4*%d + %d\n",
len, len / 4, len % 4);
*/
/*
Example:
Byte 0: 0x 43 | C | 67 / Variable amount of 'rubbish'
Byte 1: 0x 50 | P | 80 \ NOT neccessary !NULL
Byte 2: 0x 0 | . | 0 /
Byte 3: 0x 68 | h | 104 | Each attachment adds four bytes,
Byte 4: 0x ffffffc0 | . | -64 | NULL, an int, then two signed ints
Byte 5: 0x ffffff90 | . | -112 \
Byte 6: 0x 0 | . | 0 /
Byte 7: 0x 68 | h | 104 |
Byte 8: 0x ffffffc0 | . | -64 |
Byte 9: 0x ffffff92 | . | -110 \
Byte 10: 0x 0 | . | 0 /
Byte 11: 0x 68 | h | 104 |
Byte 12: 0x ffffffc0 | . | -64 |
Byte 13: 0x ffffff94 | . | -108 \
Byte 14: 0x 0 | . | 0 /
Byte 15: 0x 0 | . | 0 | Then we end with a block of
Byte 16: 0x 0 | . | 0 | four NULLs
Byte 17: 0x 0 | . | 0 \
The 'rubbish' doesn't seem to be for alignment within the pdb, AFAIKS.
*/
a->attachmentCount = (len / 4) - 1;
if (a->unknown3) {
a->unknown3length = len;
memcpy(a->unknown3, buffer, len);
SHUFFLE_BUFFER(len);
}
}
return (buffer - start);
}
int pack_VersaMail(struct VersaMail *a, char *buffer, size_t len)
{
time_t date_t;
unsigned int destlen;
char *start = buffer;
destlen = 4 + 4 + 2 + 2 + 2 + 2 + 2 + 2 + 4 + a->unknown3length;
ADD_LENGTH(a->messageUID, destlen);
ADD_LENGTH(a->to, destlen);
ADD_LENGTH(a->from, destlen);
ADD_LENGTH(a->cc, destlen);
ADD_LENGTH(a->bcc, destlen);
ADD_LENGTH(a->subject, destlen);
ADD_LENGTH(a->dateString, destlen);
ADD_LENGTH(a->body, destlen);
ADD_LENGTH(a->replyTo, destlen);
if (!buffer)
return destlen;
if (len < destlen)
return 0;
set_long(buffer, a->imapuid);
SHUFFLE_BUFFER(4);
date_t = mktime(&(a->date));
CONVERT_TIME_T_UNIX_TO_PALM(date_t);
set_long(buffer, (unsigned long) date_t);
SHUFFLE_BUFFER(4);
set_short(buffer, a->category);
SHUFFLE_BUFFER(2);
set_short(buffer, a->accountNo);
SHUFFLE_BUFFER(2);
set_short(buffer, a->unknown1);
SHUFFLE_BUFFER(2);
set_byte(buffer, a->download);
SHUFFLE_BUFFER(1);
set_byte(buffer, a->mark);
SHUFFLE_BUFFER(1);
set_short(buffer, a->unknown2);
SHUFFLE_BUFFER(2);
set_byte(buffer, a->reserved1);
set_byte(buffer + 1, ((a->reserved2 << 1) || a->read));
SHUFFLE_BUFFER(2);
set_long(buffer, a->msgSize);
SHUFFLE_BUFFER(4);
PUT_STR_FROM(a->messageUID);
PUT_STR_FROM(a->to);
PUT_STR_FROM(a->from);
PUT_STR_FROM(a->cc);
PUT_STR_FROM(a->bcc);
PUT_STR_FROM(a->subject);
PUT_STR_FROM(a->dateString);
PUT_STR_FROM(a->body);
PUT_STR_FROM(a->replyTo);
if (a->unknown3length > 0) {
memcpy(buffer, a->unknown3, a->unknown3length);
}
return (buffer - start);
}
int unpack_VersaMailAppInfo(struct VersaMailAppInfo *ai,
unsigned char *record, size_t len)
{
int i;
unsigned char *start = record;
i = unpack_CategoryAppInfo(&ai->category, record, len);
if (!i)
return i;
record += i;
len -= i;
return (record - start);
}
/*
Message provided to the maintainer of pilot-mailsync:
(the above implementation does not fully agree, BTW)
-------- Original Message --------
Subject: Re: QueueSync2.0
Date: Sat, 31 Jan 2004 03:19:28 +0900
From: Masaru matsumoto <matumoto@queuesoft.jp>
To: Jochen Garcke <jochen@garcke.de>
References: <401A7AE7.5090109@garcke.de>
Hi,
The definition of the database is to refer to the following.
But, because I analyzed it, it has the possibility to be wrong.
Write only information on VersaMail. Because there are many problems in
ClieMail.
typedef struct
{
UInt16 mu_reserved1 : 8;
UInt16 mu_reserved2 : 7;
UInt16 mu_read : 1;
} mu_MailFlagsType;
typedef struct {
UInt32 mu_version;
UInt32 mu_datetime;
UInt16 mu_category;
UInt16 mu_account_no;
UInt16 mu_rfu1;
UInt16 mu_mark;
UInt16 mu_rfu2;
mu_MailFlagsType mu_flags;
UInt32 mu_msgSize;
char mu_firstField;
} mu_MailPackedDBRecordType;
typedef mu_MailPackedDBRecordType * mu_MailPackedDBRecordPtr;
mu_msgSize =
strlen(subject)+strlen(contents)+strlen(dateString)+strlen(replyto)+4;
Field sequence :
messageUID
to
from
cc
bcc
subject
dateString
body
replayTo
*/
/* vi: set ts=8 sw=4 sts=4 noexpandtab: cin */
/* ex: set tabstop=4 expandtab: */
/* Local Variables: */
/* indent-tabs-mode: t */
/* c-basic-offset: 8 */
/* End: */