diff --git a/doc/src/FAQ/FAQ.html b/doc/src/FAQ/FAQ.html
index 0c161946e5c590859868b61f42e8c25b016b5973..4c816e7e18485ed82b4a94b53b6e397aa6737531 100644
--- a/doc/src/FAQ/FAQ.html
+++ b/doc/src/FAQ/FAQ.html
@@ -950,7 +950,7 @@ Guttman, A. "R-trees: A Dynamic Index Structure for Spatial Searching."
 Proc of the 1984 ACM SIGMOD Int'l Conf on Mgmt of Data, 45-57.<P>
 
 You can also find this paper in Stonebraker's "Readings in Database
-Systems"<P>
+Systems".<P>
 
 Built-in R-trees can handle polygons and boxes.  In theory, R-trees can
 be extended to handle higher number of dimensions.  In practice,
diff --git a/doc/src/sgml/ref/begin.sgml b/doc/src/sgml/ref/begin.sgml
index 8f7eae33c5f1e892e16ab16e514f33064ef40831..86b6620f1f7e6a90e72f93d8e38900c568fa3298 100644
--- a/doc/src/sgml/ref/begin.sgml
+++ b/doc/src/sgml/ref/begin.sgml
@@ -1,5 +1,5 @@
 <!--
-$Header: /cvsroot/pgsql/doc/src/sgml/ref/begin.sgml,v 1.11 2000/03/27 17:14:42 thomas Exp $
+$Header: /cvsroot/pgsql/doc/src/sgml/ref/begin.sgml,v 1.12 2000/10/07 14:16:01 momjian Exp $
 Postgres documentation
 -->
 
@@ -103,7 +103,7 @@ NOTICE:  BEGIN: already a transaction in progress
    and a commit is implicitly performed at the end of the statement
    (if execution was successful, otherwise a rollback is done).
    <command>BEGIN</command> initiates a user transaction in chained mode,
-   i.e. all user statements after <command>BEGIN</command> command will
+   i.e., all user statements after <command>BEGIN</command> command will
    be executed in a single transaction until an explicit 
    <xref linkend="sql-commit-title" endterm="sql-commit-title">,
    <xref linkend="sql-rollback-title" endterm="sql-rollback-title">,
diff --git a/src/backend/port/beos/Makefile b/src/backend/port/beos/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..92e11253d14cbc35172f5af2cbf9286bd10a4a43
--- /dev/null
+++ b/src/backend/port/beos/Makefile
@@ -0,0 +1,31 @@
+#-------------------------------------------------------------------------
+#
+# Makefile--
+#    Makefile for port/beos
+#
+#-------------------------------------------------------------------------
+
+top_builddir = ../../../..
+include ../../../Makefile.global
+
+INCLUDE_OPT = 
+
+CFLAGS+=$(INCLUDE_OPT)
+
+OBJS = sem.o shm.o support.o
+
+all: SUBSYS.o 
+
+SUBSYS.o: $(OBJS)
+	$(LD) $(LDREL) $(LDOUT) SUBSYS.o $(OBJS)
+
+depend dep:
+	$(CC) -MM $(INCLUDE_OPT) *.c >depend
+
+clean: 
+	rm -f SUBSYS.o $(OBJS) 
+
+ifeq (depend,$(wildcard depend))
+include depend
+endif
+
diff --git a/src/backend/port/beos/sem.c b/src/backend/port/beos/sem.c
new file mode 100644
index 0000000000000000000000000000000000000000..b67cf82f38b35a91e25df26c48f639a6a8070697
--- /dev/null
+++ b/src/backend/port/beos/sem.c
@@ -0,0 +1,224 @@
+/*-------------------------------------------------------------------------
+ *
+ * sem.c
+ *	  BeOS System V Semaphores Emulation
+ *
+ * Copyright (c) 1999-2000, Cyril VELTER
+ * 
+ *-------------------------------------------------------------------------
+ */
+
+
+#include "postgres.h"
+#include "stdio.h"
+#include "errno.h"
+#include "OS.h"
+
+// Controle d'un pool de sémaphores
+// On considere que le semId utilisé correspond bien a une area de notre adress space
+// Les informations du pool de sémaphore sont stockés dans cette area
+int semctl(int semId,int semNum,int flag,union semun semun)
+{
+	
+	// Recherche de l'adresse de base de l'area
+	int32* Address;
+	area_info info; 
+//	printf("semctl : semid  %d, semnum %d, cmd %d\n",semId,semNum,flag);
+	if (get_area_info(semId,&info)!=B_OK)
+	{
+//		printf("area not found\n");
+		errno=EINVAL;
+		return -1;
+	}
+	Address=(int32*)info.address;
+	
+	// semnum peut etre égal à 0
+	// semun.array contient la valeur de départ du sémaphore
+	
+	// si flag = set_all il faut définir la valeur du sémaphore sue semun.array
+	if (flag==SETALL)
+	{
+		long i;
+//		printf("setall %d\n",Address[0]);
+		for (i=0;i<Address[0];i++)
+		{
+			int32 cnt;
+			get_sem_count(Address[i+1],&cnt);
+//			printf("Set de ALl %d  %d = %d\n",Address[i+1],semun.array[i],cnt);
+			cnt-=semun.array[i];
+			if (cnt > 0)
+				acquire_sem_etc(Address[i+1],cnt,0,0);
+			if (cnt < 0)
+				release_sem_etc(Address[i+1],-cnt,0);
+		}
+		return 1;
+	}
+	
+	/* si flag = SET_VAL il faut définir la valeur du sémaphore sur semun.val*/
+	if (flag==SETVAL)
+	{
+		int32 cnt;
+		get_sem_count(Address[semNum+1],&cnt);
+//		printf("semctl set val id : %d val : %d = %d\n",semId,semun.val,cnt);
+		cnt-=semun.val;
+		if (cnt > 0)
+			acquire_sem_etc(Address[semNum+1],cnt,0,0);
+		if (cnt < 0)
+			release_sem_etc(Address[semNum+1],-cnt,0);
+		return 1;
+	}
+	
+	/* si flag=rm_id il faut supprimer le sémaphore*/
+	if (flag==IPC_RMID)
+	{
+		long i;
+		// Suppression des sémaphores (ils appartienent au kernel maintenant)
+		thread_info ti;
+//		printf("remove set\n");
+		get_thread_info(find_thread(NULL),&ti);
+		for (i=0;i<Address[0];i++)
+		{
+			set_sem_owner(Address[i+1],ti.team);
+			delete_sem(Address[i+1]);
+		}
+		// Il faudrait supprimer en boucle toutes les area portant le même nom
+		delete_area(semId);
+		return 1;
+	}
+	
+	/* si flag = GETNCNT il faut renvoyer le semaphore count*/
+	if (flag==GETNCNT)
+	{
+//		printf("getncnt : impossible sur BeOS\n");
+		return 0; // a faire (peut etre impossible sur Beos)
+	}
+	
+	/* si flag = GETVAL il faut renvoyer la valeur du sémaphore*/
+	if (flag==GETVAL)
+	{
+		int32 cnt;
+		get_sem_count(Address[semNum+1],&cnt);
+//		printf("semctl getval id : %d cnt : %d\n",semId,cnt);
+		return cnt;
+	}
+//	printf("semctl erreur\n");
+	return 0;
+}
+
+// L'area dans laquelle est stockée le pool est identifiée par son nom (convention à moi : SYSV_IPC_SEM : "semId)
+int semget(int semKey, int semNum, int flags)
+{
+	char Nom[50];
+	area_id parea;
+	void* Address;
+
+//	printf("semget get k: %d n: %d fl:%d\n",semKey,semNum,flags);
+	// Construction du nom que doit avoir l'area
+	sprintf(Nom,"SYSV_IPC_SEM : %d",semKey);
+
+	// Recherche de l'area
+	parea=find_area(Nom);
+
+	// L'area existe
+	if (parea!=B_NAME_NOT_FOUND)
+	{
+//		printf("area found\n");
+		// On demande une creatrion d'un pool existant : erreur
+		if ((flags&IPC_CREAT)&&(flags&IPC_EXCL))
+		{
+//			printf("creat asking exist\n");
+			errno=EEXIST;
+			return -1;
+		}
+		
+		// Clone de l'area et renvoi de son ID		
+		parea=clone_area(Nom,&Address,B_ANY_ADDRESS,B_READ_AREA | B_WRITE_AREA,parea);
+		return parea;
+	}
+	// L'area n'existe pas
+	else
+	{
+//		printf("set don't exist\n");
+		// Demande de creation
+		if (flags&IPC_CREAT)
+		{
+			int32* Address;
+			thread_info ti;
+			void* Ad;
+			long i;
+
+//			printf("create set\n");
+			// On ne peut pas creer plus de 500 semaphores dans un pool (limite tout à fait arbitraire de ma part)
+			if (semNum>500)
+			{
+				errno=ENOSPC;
+				return -1;
+			}
+					
+			// Creation de la zone de mémoire partagée
+			parea=create_area(Nom,&Ad,B_ANY_ADDRESS,4096,B_NO_LOCK,B_READ_AREA | B_WRITE_AREA);		
+			if ((parea==B_BAD_VALUE)|| (parea==B_NO_MEMORY)||(parea==B_ERROR))
+			{
+				errno=ENOMEM;
+				return -1;
+			}
+			Address=(int32*)Ad;
+			Address[0]=semNum;
+			for (i=1;i<=Address[0];i++)
+			{
+				// Creation des sémaphores 1 par 1
+				Address[i]=create_sem(0,Nom);
+				
+				if ((Address[i]==B_BAD_VALUE)|| (Address[i]==B_NO_MEMORY)||(Address[i]==B_NO_MORE_SEMS))
+				{
+					errno=ENOMEM;
+					return -1;
+				}
+			}
+
+//			printf("returned %d\n",parea);
+			return parea;
+		}
+		// Le pool n'existe pas et pas de demande de création
+		else
+		{
+//			printf("set does not exist no creat requested\n");
+			errno=ENOENT;
+			return -1;
+		}
+	}
+}
+
+// Opération sur le pool de sémaphores
+int semop(int semId, struct sembuf *sops, int nsops)
+{
+	// Recherche de l'adresse du pool
+	int32* Address;
+   	area_info info; 
+	long i;
+
+//	printf("semop id : %d n: %d\n",semId,sops->sem_op);
+	get_area_info(semId,&info);
+	Address=(int32*)info.address;
+	if ((semId==B_BAD_VALUE)||(semId==B_NO_MEMORY)||(semId==B_ERROR))
+	{
+		errno=EINVAL;
+		return -1;
+	}
+
+	// Execution de l'action
+	for(i=0;i<nsops;i++)
+	{
+
+//		printf("semid %d, n %d\n",Address[sops[i].sem_num+1],sops[i].sem_op);
+		if (sops[i].sem_op < 0)
+		{
+			acquire_sem_etc(Address[sops[i].sem_num+1],-sops[i].sem_op,0,0);
+		}
+		if (sops[i].sem_op > 0)
+		{
+			release_sem_etc(Address[sops[i].sem_num+1],sops[i].sem_op,0);
+		}
+	}
+	return 0;
+}
diff --git a/src/backend/port/beos/shm.c b/src/backend/port/beos/shm.c
new file mode 100644
index 0000000000000000000000000000000000000000..91e6756b738ba95ec91f3b5c44e15a566fefa3f1
--- /dev/null
+++ b/src/backend/port/beos/shm.c
@@ -0,0 +1,112 @@
+/*-------------------------------------------------------------------------
+ *
+ * shm.c
+ *	  BeOS System V Shared Memory Emulation
+ *
+ * Copyright (c) 1999-2000, Cyril VELTER
+ * 
+ *-------------------------------------------------------------------------
+ */
+
+#include "postgres.h"
+#include "stdio.h"
+#include "OS.h"
+
+// Detachement d'une zone de mémoire partagée
+// On detruit le clone de l'area dans notre adress-space
+int shmdt(char* shmaddr)
+{
+	// Recherche de l'id de l'area présente à cette adresse
+	area_id s;
+	s=area_for(shmaddr);
+//	printf("detach area %d\n",s);
+	
+	// Suppression de l'area
+	return delete_area(s);
+}
+
+// Attachement à une zone de mémoire partagée
+// L'area doit bien partie de notre adress-space et on retourne directement l'adress
+int* shmat(int memId,int m1,int m2)
+{
+//	printf("shmat %d %d %d\n",memId,m1,m2);
+
+	// Lecture de notre team_id
+	thread_info thinfo;
+	team_info teinfo;
+	area_info ainfo; 
+	
+	get_thread_info(find_thread(NULL),&thinfo);
+	get_team_info(thinfo.team,&teinfo);
+	
+	// Lecture du teamid de l'area
+	if (get_area_info(memId,&ainfo)!=B_OK)
+		printf("AREA %d Invalide\n",memId);
+	
+	if (ainfo.team==teinfo.team)
+	{
+		//retour de l'adresse
+//		printf("attach area %d add %d\n",memId,ainfo.address);
+		return (int*)ainfo.address;
+	}	
+	else
+	{
+		// Clone de l'area
+		area_id narea;
+		narea = clone_area(ainfo.name,&(ainfo.address),B_CLONE_ADDRESS,B_READ_AREA | B_WRITE_AREA,memId);	
+		get_area_info(narea,&ainfo);	
+//		printf("attach area %d in %d add %d\n",memId,narea,ainfo.address);
+		return (int*)ainfo.address;
+	}
+}
+
+// Utilisé uniquement pour supprimer une zone de mémoire partagée
+// On fait la meme chose que le detach mais avec un id direct
+int shmctl(int shmid,int flag, struct shmid_ds* dummy)
+{
+//	printf("shmctl %d %d \n",shmid,flag);
+	delete_area(shmid);
+	return 0;
+}
+
+// Recupération d'une area en fonction de sa référence
+// L'area source est identifiée par son nom (convention à moi : SYSV_IPC_SHM : "memId)
+int shmget(int memKey,int size,int flag)
+{
+	int32 n_size;
+	char nom[50];
+	area_id parea;
+	void* Address;
+	area_id a;
+	
+	n_size=((size/4096)+1)*4096;
+
+//	printf("shmget %d %d %d %d\n",memKey,size,flag,nsize);
+
+	// Determination du nom que doit avoir l'area
+	sprintf(nom,"SYSV_IPC_SHM : %d",memKey);
+
+
+	// Recherche de cette area
+	parea=find_area(nom);
+	
+	// L'area existe
+	if (parea!=B_NAME_NOT_FOUND)
+	{
+//		printf("area found\n");
+		return parea;
+	}	
+
+	// L'area n'existe pas et on n'en demande pas la création : erreur
+	if (flag==0)
+	{
+//		printf("area %s not found\n",nom);
+		return -1;
+	}
+	
+	// L'area n'existe pas mais on demande sa création
+	a=create_area(nom,&Address,B_ANY_ADDRESS,n_size,B_NO_LOCK,B_READ_AREA | B_WRITE_AREA);		
+//	printf("area %s : %d created addresse %d\n",nom,a,Address);
+	return a;
+}
+
diff --git a/src/backend/port/beos/support.c b/src/backend/port/beos/support.c
new file mode 100644
index 0000000000000000000000000000000000000000..d517dd7af37d144a62fdf244c942a6c523ff58f5
--- /dev/null
+++ b/src/backend/port/beos/support.c
@@ -0,0 +1,258 @@
+/*-------------------------------------------------------------------------
+ *
+ * support.c
+ *	  BeOS Support functions
+ *
+ * Copyright (c) 1999-2000, Cyril VELTER
+ * 
+ *-------------------------------------------------------------------------
+ */
+
+#include "postgres.h"
+
+/* Support Globals */
+char* self_binary=NULL;
+port_id beos_dl_port_in=0;
+port_id beos_dl_port_out=0;
+sem_id beos_shm_sem;
+
+image_id beos_dl_open(char * filename)
+{
+	image_id im;
+
+	/* Start the support server */
+	if (self_binary==NULL)
+	{
+		/* Can't start support server without binary name */		
+		elog(NOTICE, "Error loading BeOS support server : can't find binary");				
+		return B_ERROR;
+	}
+	else
+	{
+		/* If a port doesn't exist, lauch support server */ 
+		if ((beos_dl_port_in<=0)||(beos_dl_port_out<=0))
+		{
+			/* Create communication port */
+			beos_dl_port_in=create_port(50,"beos_support_in");
+			beos_dl_port_out=create_port(50,"beos_support_in");
+
+
+			if ((beos_dl_port_in<=0)||(beos_dl_port_out<=0))
+			{
+				elog(NOTICE, "Error loading BeOS support server : can't create communication ports");				
+				return B_ERROR;
+			}
+			else
+			{
+				char Cmd[4000]; 
+			
+				/* Build arg list */
+				sprintf(Cmd,"%s -beossupportserver %d %d &",self_binary,(int)beos_dl_port_in,(int)beos_dl_port_out);
+	
+				/* Lauch process */
+				system(Cmd);
+			}
+		}
+	}
+	
+	/* Add-on loading */
+	
+	/* Send command '1' (load) to the support server */
+	write_port(beos_dl_port_in,1,filename,strlen(filename)+1);
+	
+	/* Read Object Id */
+	read_port(beos_dl_port_out,&im,NULL,0);
+
+	/* Checking integrity */
+	if (im<0)
+	{	
+		elog(NOTICE, "Can't load this add-on ");
+		return B_ERROR;	
+	}
+	else
+	{
+		/* Map text and data segment in our address space */
+		char datas[4000];
+		int32 area;
+		int32 resu;
+		void* add;
+	
+		/* read text segment id and address */
+		read_port(beos_dl_port_out,&area,datas,4000);
+		read_port(beos_dl_port_out,(void*)&add,datas,4000);
+		/* map text segment in our address space */
+		resu=clone_area(datas,&add,B_EXACT_ADDRESS,B_READ_AREA|B_WRITE_AREA,area);
+		if (resu<0)
+		{
+			/* If we can't map, we are in reload case */
+			/* delete the mapping */
+			resu=delete_area(area_for(add));
+			/* Remap */
+			resu=clone_area(datas,&add,B_EXACT_ADDRESS,B_READ_AREA|B_WRITE_AREA,area);
+			if (resu<0)
+			{
+				elog(NOTICE, "Can't load this add-on : map text error");
+			}
+		}
+		
+		/* read text segment id and address */
+		read_port(beos_dl_port_out,&area,datas,4000);
+		read_port(beos_dl_port_out,(void*)&add,datas,4000);
+		/* map text segment in our address space */
+		resu=clone_area(datas,&add,B_EXACT_ADDRESS,B_READ_AREA|B_WRITE_AREA,area);
+		if (resu<0)
+		{
+			/* If we can't map, we are in reload case */
+			/* delete the mapping */
+			resu=delete_area(area_for(add));
+			/* Remap */
+			resu=clone_area(datas,&add,B_EXACT_ADDRESS,B_READ_AREA|B_WRITE_AREA,area);
+			if (resu<0)
+			{
+				elog(NOTICE, "Can't load this add-on : map data error");
+			}
+		}
+		
+		return im;
+	}
+}
+
+status_t beos_dl_close(image_id im)
+{
+	/* unload add-on */
+	int32 resu;
+	write_port(beos_dl_port_in,2,&im,4);
+	read_port(beos_dl_port_out,&resu,NULL,0);
+	return resu;
+}
+
+/* Main support server loop */
+
+void beos_startup(int argc,char** argv)
+{
+	if (strlen(argv[0]) >= 10 && !strcmp(argv[0] + strlen(argv[0]) - 10, "postmaster"))
+	{
+		/* Shared memory cloning protection semaphore */
+		beos_shm_sem=create_sem(1,"beos_shm_sem");	
+	}
+
+	if (argc > 1 && strcmp(argv[1], "-beossupportserver") == 0)
+	{
+		port_id port_in;
+		port_id port_out;
+		
+		/* Get back port ids from arglist */
+		sscanf(argv[2],"%d",(int*)(&port_in));
+		sscanf(argv[3],"%d",(int*)(&port_out));
+			
+		/* Main server loop */
+		for (;;)
+		{ 
+			int32 opcode=0;
+			char datas[4000];
+	
+			/* Wait for a message from the backend :
+			1 : load a shared object 
+			2 : unload a shared object
+			any other : exit support server */
+			read_port(port_in,&opcode,datas,4000);
+	
+			switch(opcode)
+			{
+				image_id addon;
+				image_info info_im;
+				area_info info_ar;
+	
+				/* Load Add-On */
+				case 1 :
+		
+					/* Load shared object */
+					addon=load_add_on(datas);
+		
+					/* send back the shared object Id */
+					write_port(port_out,addon,NULL,0);
+					
+					/* Get Shared Object infos */
+					get_image_info(addon,&info_im);
+	
+					/* get text segment info */
+					get_area_info(area_for(info_im.text),&info_ar);
+					/* Send back area_id of text segment */
+					write_port(port_out,info_ar.area,info_ar.name,strlen(info_ar.name)+1);
+					/* Send back real address of text segment */
+					write_port(port_out,(int)info_ar.address,info_ar.name,strlen(info_ar.name)+1);
+			
+					
+					/* get data segment info */
+					get_area_info(area_for(info_im.data),&info_ar);
+					/* Send back area_id of data segment */
+					write_port(port_out,info_ar.area,info_ar.name,strlen(info_ar.name)+1);
+					/* Send back real address of data segment */
+					write_port(port_out,(int)info_ar.address,info_ar.name,strlen(info_ar.name)+1);
+				break;
+				/* UnLoad Add-On */
+				case 2 :
+					/* Unload shared object and send back the result of the operation */
+					write_port(port_out,unload_add_on(*((int*)(datas))),NULL,0);
+				break;
+				/* Cleanup and exit */
+				default:
+					/* Free system resources */
+					delete_port(port_in);
+					delete_port(port_out);
+					/* Exit */
+					exit(0);
+				break;
+			}
+		}
+		/* Never be there */
+		exit(1);
+	}
+}
+
+
+void beos_backend_startup(char * binary)
+{
+	team_id ct;
+	thread_info inft;
+	char nom[50];
+	char nvnom[50];
+	area_info inf;
+	int32 cook=0;
+
+	/* remember full path binary name to load dl*/
+	self_binary=strdup(binary);
+
+	/* find the current team */
+	get_thread_info(find_thread(NULL),&inft);
+	ct=inft.team;
+	
+	/* find all area with a name begining by pgsql and destroy / clone then */
+
+	/* This operation must be done by only one backend at a time */
+	if(acquire_sem(beos_shm_sem)==B_OK)
+	{
+		while (get_next_area_info(0, &cook, &inf) == B_OK)
+		{
+			strcpy(nom,inf.name);
+			strcpy(nvnom,inf.name);
+			nom[9]=0;
+			nvnom[5]='i';
+			if (!strcmp(nom,"SYSV_IPC_"))
+			{
+				void* add;
+				area_id ar;
+				add=inf.address;
+				delete_area(inf.area);
+				ar=find_area(inf.name);
+				clone_area(nvnom,&add,B_CLONE_ADDRESS,B_READ_AREA|B_WRITE_AREA,ar);
+			}
+		} 
+		release_sem(beos_shm_sem);
+	}
+	else
+	{
+		/* Fatal error, exiting with error */
+		exit(1);
+	}
+}