605 lines
15 KiB
C
605 lines
15 KiB
C
/*
|
|
* CDE - Common Desktop Environment
|
|
*
|
|
* Copyright (c) 1993-2012, The Open Group. All rights reserved.
|
|
*
|
|
* These libraries and programs are free software; you can
|
|
* redistribute them and/or modify them under the terms of the GNU
|
|
* Lesser General Public License as published by the Free Software
|
|
* Foundation; either version 2 of the License, or (at your option)
|
|
* any later version.
|
|
*
|
|
* These libraries and programs are distributed in the hope that
|
|
* they will be useful, but WITHOUT ANY WARRANTY; without even the
|
|
* implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
* PURPOSE. See the GNU Lesser General Public License for more
|
|
* details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with these librararies and programs; if not, write
|
|
* to the Free Software Foundation, Inc., 51 Franklin Street, Fifth
|
|
* Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
/* *
|
|
* (c) Copyright 1993, 1994 Hewlett-Packard Company *
|
|
* (c) Copyright 1993, 1994 International Business Machines Corp. *
|
|
* (c) Copyright 1993, 1994 Sun Microsystems, Inc. *
|
|
* (c) Copyright 1993, 1994 Novell, Inc. *
|
|
*/
|
|
/*Add a string to the XA_RESOURCE_MANAGER*/
|
|
/*
|
|
* COPYRIGHT 1987
|
|
* DIGITAL EQUIPMENT CORPORATION
|
|
* MAYNARD, MASSACHUSETTS
|
|
* ALL RIGHTS RESERVED.
|
|
*
|
|
* THE INFORMATION IN THIS SOFTWARE IS SUBJECT TO CHANGE WITHOUT NOTICE AND
|
|
* SHOULD NOT BE CONSTRUED AS A COMMITMENT BY DIGITAL EQUIPMENT CORPORATION.
|
|
* DIGITAL MAKES NO REPRESENTATIONS ABOUT THE SUITABILITY OF THIS SOFTWARE FOR
|
|
* ANY PURPOSE. IT IS SUPPLIED "AS IS" WITHOUT EXPRESS OR IMPLIED WARRANTY.
|
|
*
|
|
* IF THE SOFTWARE IS MODIFIED IN A MANNER CREATING DERIVATIVE COPYRIGHT RIGHTS,
|
|
* APPROPRIATE LEGENDS MAY BE PLACED ON THE DERIVATIVE WORK IN ADDITION TO THAT
|
|
* SET FORTH ABOVE.
|
|
*
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software and its
|
|
* documentation for any purpose and without fee is hereby granted, provided
|
|
* that the above copyright notice appear in all copies and that both that
|
|
* copyright notice and this permission notice appear in supporting
|
|
* documentation, and that the name of Digital Equipment Corporation not be
|
|
* used in advertising or publicity pertaining to distribution of the software
|
|
* without specific, written prior permission.
|
|
*/
|
|
|
|
/*Lifted from xrdb(1X)*/
|
|
/* -*-C-*-
|
|
*******************************************************************************
|
|
*
|
|
* File: addToResource.c
|
|
* RCS: $TOG: addToRes.c /main/8 1999/10/14 16:01:17 mgreess $
|
|
* Description: Source code for adding strings to RESOURCE_PROPERTY on
|
|
default root window
|
|
* Author: DEC, Robert Williams
|
|
* Created: Thu Apr 26 14:42:08 PDT 1990
|
|
* Modified: Kim Dronesen
|
|
* Language: C
|
|
* Package: N/A
|
|
* Status: Experimental (Do Not Distribute)
|
|
*
|
|
* (C) Copyright 1990, Hewlett-Packard, all rights reserved.
|
|
*
|
|
*******************************************************************************
|
|
*/
|
|
/*$TOG: addToRes.c /main/8 1999/10/14 16:01:17 mgreess $ */
|
|
|
|
#include <X11/Xlib.h>
|
|
#include <X11/Xatom.h>
|
|
#include <X11/Xos.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
#include <Dt/DtP.h>
|
|
#include <Dt/SessionM.h>
|
|
#include "DtSvcLock.h"
|
|
|
|
/****************************************/
|
|
/* Global variables */
|
|
/****************************************/
|
|
|
|
typedef struct _Entry {
|
|
char *tag, *value;
|
|
int lineno;
|
|
Bool usable;
|
|
} Entry;
|
|
typedef struct _Buffer {
|
|
char *buff;
|
|
int room, used;
|
|
Bool freebuff;
|
|
} Buffer;
|
|
typedef struct _Entries {
|
|
Entry *entry;
|
|
int room, used;
|
|
} Entries;
|
|
|
|
#define INIT_BUFFER_SIZE 10000
|
|
#define INIT_ENTRY_SIZE 500
|
|
|
|
|
|
/******** Public Function Declarations ********/
|
|
|
|
static Buffer * _DtAllocBuffer(const char **buff);
|
|
static void _DtFreeBuffer(Buffer *b);
|
|
static void _DtAppendToBuffer(
|
|
register Buffer *b,
|
|
char *str,
|
|
register int len) ;
|
|
static void _DtAppendEntryToBuffer(
|
|
register Buffer *buffer,
|
|
Entry entry) ;
|
|
|
|
static Entries * _DtAllocEntries( void) ;
|
|
static void _DtFreeEntries( Entries *) ;
|
|
static void _DtAddEntry(
|
|
register Entries *e,
|
|
Entry entry) ;
|
|
static int _DtCompareEntries(
|
|
Entry *e1,
|
|
Entry *e2) ;
|
|
static char * _DtFindFirst(
|
|
register char *string,
|
|
register char dest) ;
|
|
static void _DtGetEntries(
|
|
register Entries *entries,
|
|
Buffer *buff,
|
|
int dosort) ;
|
|
static void _DtMergeEntries(
|
|
Buffer *buffer,
|
|
Entries new,
|
|
Entries old) ;
|
|
static void _DtAddToResProp(
|
|
Display *dpy,
|
|
unsigned int id,
|
|
Entries db) ;
|
|
static void _getWinProp(
|
|
Display *dpy,
|
|
unsigned int id,
|
|
Window *win,
|
|
Atom *prop);
|
|
|
|
/******** End Public Function Declarations ********/
|
|
|
|
/****************************************/
|
|
/*The meat*/
|
|
/****************************************/
|
|
|
|
static Buffer *
|
|
_DtAllocBuffer(
|
|
const char **buff )
|
|
{
|
|
Buffer *b = (Buffer *)malloc(sizeof(Buffer));
|
|
b->room = INIT_BUFFER_SIZE;
|
|
b->buff = buff ? (char*) *buff :
|
|
(char *)malloc(INIT_BUFFER_SIZE*sizeof(char));
|
|
b->used = buff && *buff ? strlen(*buff) : 0;
|
|
b->freebuff = buff ? False : True;
|
|
return(b);
|
|
}
|
|
|
|
static void
|
|
_DtFreeBuffer(
|
|
Buffer *b)
|
|
{
|
|
if (b->freebuff == True) free(b->buff);
|
|
free(b);
|
|
}
|
|
|
|
static void
|
|
_DtAppendToBuffer(
|
|
register Buffer *b,
|
|
char *str,
|
|
register int len )
|
|
{
|
|
while (b->used + len > b->room) {
|
|
b->buff = (char *)realloc(b->buff, 2*b->room*(sizeof(char)));
|
|
b->room *= 2;
|
|
}
|
|
strncpy(b->buff + b->used, str, len);
|
|
b->used += len;
|
|
}
|
|
|
|
static Entries *
|
|
_DtAllocEntries( void )
|
|
{
|
|
Entries *e = (Entries *)malloc(sizeof(Entries));
|
|
e->room = INIT_ENTRY_SIZE;
|
|
e->used = 0;
|
|
e->entry = (Entry *)malloc(INIT_ENTRY_SIZE*sizeof(Entry));
|
|
return(e);
|
|
}
|
|
|
|
static void
|
|
_DtFreeEntries(
|
|
Entries *e)
|
|
{
|
|
int n = 0;
|
|
|
|
while (n < e->used)
|
|
{
|
|
free(e->entry[n].tag);
|
|
free(e->entry[n].value);
|
|
n++;
|
|
}
|
|
free(e->entry);
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_DtAddEntry(
|
|
register Entries *e,
|
|
Entry entry )
|
|
{
|
|
register int n;
|
|
|
|
for (n = 0; n < e->used; n++)
|
|
{
|
|
if (strcmp(e->entry[n].tag, entry.tag) == 0)
|
|
{ /* overwrite old entry - free its memory first*/
|
|
free(e->entry[n].tag);
|
|
free(e->entry[n].value);
|
|
e->entry[n] = entry;
|
|
return ; /* ok to leave, now there's only one of each tag in db */
|
|
}
|
|
}
|
|
|
|
if (e->used == e->room) {
|
|
e->entry = (Entry *)realloc(e->entry, 2*e->room*(sizeof(Entry)));
|
|
e->room *= 2;
|
|
}
|
|
entry.usable = True;
|
|
e->entry[e->used++] = entry;
|
|
}
|
|
|
|
static int
|
|
_DtCompareEntries(
|
|
Entry *e1,
|
|
Entry *e2 )
|
|
{
|
|
return strcmp(e1->tag, e2->tag);
|
|
}
|
|
|
|
static void
|
|
_DtAppendEntryToBuffer(
|
|
register Buffer *buffer,
|
|
Entry entry )
|
|
{
|
|
_DtAppendToBuffer(buffer, entry.tag, strlen(entry.tag));
|
|
_DtAppendToBuffer(buffer, ":\t", 2);
|
|
_DtAppendToBuffer(buffer, entry.value, strlen(entry.value));
|
|
_DtAppendToBuffer(buffer, "\n", 1);
|
|
}
|
|
|
|
static char *
|
|
_DtFindFirst(
|
|
register char *string,
|
|
register char dest )
|
|
{
|
|
int len;
|
|
|
|
for (;;) {
|
|
if((len = mblen(string, MB_CUR_MAX)) > 1) {
|
|
string += len;
|
|
continue;
|
|
}
|
|
if (*string == '\0')
|
|
return NULL;
|
|
if (*string == '\\') {
|
|
if (*++string == '\0')
|
|
return NULL;
|
|
}
|
|
else if (*string == dest)
|
|
return string;
|
|
string++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
_DtGetEntries(
|
|
register Entries *entries,
|
|
Buffer *buff,
|
|
int dosort )
|
|
{
|
|
register char *line, *colon, *temp, *str, *temp2;
|
|
Entry entry;
|
|
register int length;
|
|
int lineno = 0;
|
|
|
|
str = buff->buff;
|
|
if (!str) return;
|
|
for (; str < buff->buff + buff->used; str = line + 1)
|
|
{
|
|
line = _DtFindFirst(str, '\n');
|
|
lineno++;
|
|
if (line == NULL)
|
|
break;
|
|
if (str[0] == '!')
|
|
continue;
|
|
if (str[0] == '\n')
|
|
continue;
|
|
if (str[0] == '#')
|
|
{
|
|
int dummy;
|
|
if (sscanf (str, "# %d", &dummy) == 1) lineno = dummy - 1;
|
|
continue;
|
|
}
|
|
for (temp = str;
|
|
*temp && *temp != '\n' && isascii(*temp) && isspace((u_char)*temp);
|
|
temp++) ;
|
|
if (!*temp || *temp == '\n') continue;
|
|
|
|
colon = _DtFindFirst(str, ':');
|
|
if (colon == NULL)
|
|
break;
|
|
if (colon > line)
|
|
{
|
|
line[0] = '\0';
|
|
fprintf (stderr,
|
|
"%s: colon missing on line %d, ignoring entry \"%s\"\n",
|
|
"dtprefs", lineno, str);
|
|
continue;
|
|
}
|
|
|
|
temp2 = str;
|
|
while (temp2[0] == ' ' || temp2[0] == '\t')
|
|
{
|
|
temp2++;
|
|
}
|
|
temp = (char *)malloc((length = colon - temp2) + 1);
|
|
strncpy(temp, temp2, length);
|
|
temp[length] = '\0';
|
|
while (temp[length-1] == ' ' || temp[length-1] == '\t')
|
|
temp[--length] = '\0';
|
|
entry.tag = temp;
|
|
|
|
|
|
temp2 = colon + 1;
|
|
while (temp2[0] == ' ' || temp2[0] == '\t')
|
|
{
|
|
temp2++;
|
|
}
|
|
temp = (char *)malloc((length = line - temp2) + 1);
|
|
strncpy(temp, temp2, length);
|
|
temp[length] = '\0';
|
|
entry.value = temp;
|
|
entry.lineno = lineno;
|
|
|
|
_DtAddEntry(entries, entry);
|
|
}
|
|
|
|
if (dosort && (entries->used > 0))
|
|
qsort(entries->entry, entries->used, sizeof(Entry),
|
|
(int (*)(const void *, const void *))_DtCompareEntries);
|
|
}
|
|
|
|
static void
|
|
_DtMergeEntries(
|
|
Buffer *buffer,
|
|
Entries new,
|
|
Entries old )
|
|
{
|
|
int n, o, cmp;
|
|
|
|
buffer->used = 0;
|
|
n = o = 0;
|
|
while ((n < new.used) && (o < old.used))
|
|
{
|
|
cmp = strcmp(new.entry[n].tag, old.entry[o].tag);
|
|
if (cmp > 0)
|
|
{
|
|
_DtAppendEntryToBuffer(buffer, old.entry[o]);
|
|
o++;
|
|
}
|
|
else
|
|
{
|
|
_DtAppendEntryToBuffer(buffer, new.entry[n]);
|
|
n++;
|
|
if (cmp == 0)
|
|
{
|
|
o++;
|
|
}
|
|
}
|
|
}
|
|
|
|
while (n < new.used)
|
|
{
|
|
_DtAppendEntryToBuffer(buffer, new.entry[n]);
|
|
n++;
|
|
}
|
|
while (o < old.used)
|
|
{
|
|
_DtAppendEntryToBuffer(buffer, old.entry[o]);
|
|
o++;
|
|
}
|
|
|
|
_DtAppendToBuffer(buffer, "\0", 1);
|
|
}
|
|
|
|
static void
|
|
_getWinProp(
|
|
Display *dpy,
|
|
unsigned int id,
|
|
Window *win,
|
|
Atom *prop)
|
|
{
|
|
static Bool init = True;
|
|
static Window winprop;
|
|
static Atom xa_resmgr;
|
|
static Atom xa_prefs;
|
|
|
|
_DtSvcProcessLock();
|
|
if (init == True)
|
|
{
|
|
winprop = XRootWindow(dpy, 0);
|
|
xa_resmgr = XA_RESOURCE_MANAGER;
|
|
xa_prefs = XInternAtom (dpy, _XA_DT_SM_PREFERENCES, False);
|
|
init = False;
|
|
}
|
|
_DtSvcProcessUnlock();
|
|
|
|
*win = winprop;
|
|
*prop = id == _DT_ATR_RESMGR ? xa_resmgr : xa_prefs;
|
|
}
|
|
|
|
|
|
static void
|
|
_DtAddToResProp(
|
|
Display *dpy,
|
|
unsigned int id,
|
|
Entries db)
|
|
{
|
|
char *xdefs;
|
|
Buffer *oldBuffer, *newBuffer;
|
|
Entries *oldDB;
|
|
int defStatus;
|
|
Atom actualType;
|
|
int actualFormat;
|
|
unsigned long nitems, leftover;
|
|
Window win;
|
|
Atom prop;
|
|
|
|
/*
|
|
* Get window and property
|
|
*/
|
|
_getWinProp(dpy, id, &win, &prop);
|
|
if (win == (Window)0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Get resource database from specified window and property.
|
|
*/
|
|
defStatus = XGetWindowProperty(dpy, win,
|
|
prop, 0L,
|
|
100000000L,False,XA_STRING,&actualType,
|
|
&actualFormat,&nitems,&leftover,
|
|
(unsigned char**) &xdefs);
|
|
|
|
|
|
/*
|
|
* Allocate oldBuffer and init from resource database string
|
|
*/
|
|
oldBuffer = _DtAllocBuffer((const char**) &xdefs);
|
|
|
|
/*
|
|
* Allocate oldDB
|
|
*/
|
|
oldDB = _DtAllocEntries();
|
|
|
|
/*
|
|
* Convert oldBuffer to oldDB.
|
|
*/
|
|
_DtGetEntries(oldDB, oldBuffer, 1);
|
|
|
|
/*
|
|
* Init empty newBuffer, then populate by merging db into oldDB.
|
|
*/
|
|
newBuffer = _DtAllocBuffer(NULL);
|
|
_DtMergeEntries(newBuffer, db, *oldDB);
|
|
|
|
/*
|
|
* Finally, store newBuffer into resource database.
|
|
*/
|
|
XChangeProperty (dpy, win, prop,
|
|
XA_STRING, 8, PropModeReplace,
|
|
(unsigned char *)newBuffer->buff, newBuffer->used);
|
|
|
|
XSync(dpy, False);
|
|
|
|
/*
|
|
* Free buffer memory
|
|
*/
|
|
if (oldBuffer->buff) XFree(oldBuffer->buff);
|
|
_DtFreeBuffer(oldBuffer);
|
|
_DtFreeBuffer(newBuffer);
|
|
_DtFreeEntries(oldDB);
|
|
}
|
|
|
|
char *
|
|
_DtGetResString(
|
|
Display *dpy,
|
|
unsigned int id)
|
|
{
|
|
char *xdefs;
|
|
Buffer *oldBuffer, *newBuffer;
|
|
Entries *oldDB;
|
|
int defStatus;
|
|
Atom actualType;
|
|
int actualFormat;
|
|
unsigned long nitems, leftover;
|
|
Window win;
|
|
Atom prop;
|
|
|
|
/*
|
|
* Get window and property
|
|
*/
|
|
_getWinProp(dpy, id, &win, &prop);
|
|
if (win == (Window)0)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Get resource database from specified window and property.
|
|
*/
|
|
defStatus = XGetWindowProperty(dpy, win,
|
|
prop, 0L,
|
|
100000000L,False,XA_STRING,&actualType,
|
|
&actualFormat,&nitems,&leftover,
|
|
(unsigned char**) &xdefs);
|
|
|
|
return(xdefs);
|
|
}
|
|
|
|
void
|
|
_DtAddToResource(
|
|
Display *dpy,
|
|
const char *data )
|
|
{
|
|
_DtAddResString( dpy, data, _DT_ATR_RESMGR|_DT_ATR_PREFS);
|
|
}
|
|
|
|
void
|
|
_DtAddResString(
|
|
Display *dpy,
|
|
const char *data,
|
|
unsigned int flags)
|
|
{
|
|
char *xdefs;
|
|
int i;
|
|
Buffer *buffer;
|
|
Entries *newDB;
|
|
int defStatus;
|
|
Atom actualType;
|
|
int actualFormat;
|
|
unsigned long nitems, leftover;
|
|
|
|
if((data == NULL) || (*data == '\0'))
|
|
{
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Init buffer with input data
|
|
*/
|
|
buffer = _DtAllocBuffer(&data);
|
|
|
|
/*
|
|
* Init, then populate, newDB from buffer
|
|
*/
|
|
newDB = _DtAllocEntries();
|
|
_DtGetEntries(newDB, buffer, 1);
|
|
|
|
if (flags & _DT_ATR_RESMGR)
|
|
{
|
|
/*
|
|
* Merge newDB into RESOURCE_MANAGER
|
|
*/
|
|
_DtAddToResProp(dpy, _DT_ATR_RESMGR, *newDB);
|
|
}
|
|
|
|
if (flags & _DT_ATR_PREFS)
|
|
{
|
|
/*
|
|
* Merge newDB into _DT_SM_PREFERENCES
|
|
*/
|
|
_DtAddToResProp(dpy, _DT_ATR_PREFS, *newDB);
|
|
}
|
|
|
|
/*
|
|
* Free objects
|
|
*/
|
|
_DtFreeBuffer(buffer);
|
|
_DtFreeEntries(newDB);
|
|
}
|