From: pottier@clipper.ens.fr (Francois Pottier)
Subject: csmp-digest-v3-017
Date: Wed, 20 Apr 94 13:09:01 MET DST

C.S.M.P. Digest             Wed, 20 Apr 94       Volume 3 : Issue 17
 
Today's Topics:
 
        'aete' and AppleScript
        Help on recursive read of directory catalog..?
        How can I display TIFF?
        Inverting a button in a dialog
        Looking for styled TE replacement?
        Mounting AFPServer volume...
        PPC ThreadManager w-CodeWarrior
        RJW: Retrieving application name from OSType Creator
        Range of OSErr's for private use?
        Simple Q: Assigning char * to char []. How?
        [Q] Validity of a memory address
        textedit bounds



The Comp.Sys.Mac.Programmer Digest is moderated by Francois Pottier
(pottier@clipper.ens.fr).
 
The digest is a collection of article threads from the internet newsgroup
comp.sys.mac.programmer.  It is designed for people who read c.s.m.p. semi-
regularly and want an archive of the discussions.  If you don't know what a
newsgroup is, you probably don't have access to it.  Ask your systems
administrator(s) for details.  If you don't have access to news, you may
still be able to post messages to the group by using a mail server like
anon.penet.fi (mail help@anon.penet.fi for more information).
 
Each issue of the digest contains one or more sets of articles (called
threads), with each set corresponding to a 'discussion' of a particular
subject.  The articles are not edited; all articles included in this digest
are in their original posted form (as received by our news server at
nef.ens.fr).  Article threads are not added to the digest until the last
article added to the thread is at least two weeks old (this is to ensure that
the thread is dead before adding it to the digest).  Article threads that
consist of only one message are generally not included in the digest.

The digest is officially distributed by two means, by email and ftp.

If you want to receive the digest by mail, send email to listserv@ens.fr
with no subject and one of the following commands as body:
    help		                Sends you a summary of commands
    subscribe csmp-digest Your Name	Adds you to the mailing list
    signoff csmp-digest			Removes you from the list
Once you have subscribed, you will automatically receive each new
issue as it is created.

The official ftp info is //ftp.dartmouth.edu/pub/csmp-digest.
Questions related to the ftp site should be directed to
scott.silver@dartmouth.edu. Currently no previous volumes of the CSMP
digest are available there.

Also, the digests are available to WAIS users as comp.sys.mac.programmer.src.


-------------------------------------------------------

>From "Jared M. Oberhaus" <oberhaus+@CMU.EDU>
Subject: 'aete' and AppleScript
Date: Tue, 29 Mar 1994 03:54:34 -0500
Organization: Senior, Electrical and Computer Engineering, Carnegie Mellon, Pittsburgh, PA

OK, I think I've figured out that the aete resource is the Dictionary
for translating raw AppleEvent formats into plain English-like commands.

But, where is the Template resource for ResEdit, so I can create and
modify this resource?

And if it doesn't exist yet, how do all you other guys do it? Not the
hard way, I hope...

Help!!

ps-I saw the demo of Resorcerer and noticed that it supports aete
resources, and looked around, and I supposed I could figure out how to
do an aete resoure by looking at the hex offsets, etc., but I don't
exactly feel like doing that!

pps-If you could, when you respond, could you respond to All (that is,
put me on the adressee list) instead of just Readers, that way, it'll be
easier than looking through five hundred messages every day... Thanks!


- ---------------------------------------------------------------------------
Jared M. Oberhaus                                           oberhaus+@CMU.EDU
Electrical and Computer Engineering                             Class of 1994
Carnegie Mellon University
- ---------------------------------------------------------------------------
"We are upping our standards,...so up yours."  
		- Pat Paulsen for President, 1988


+++++++++++++++++++++++++++

>From ldo@waikato.ac.nz (Lawrence D'Oliveiro, Waikato University)
Date: 30 Mar 94 11:19:56 +1300
Organization: University of Waikato, Hamilton, New Zealand

In article <AhZyn_y00iV4E0fEpd@andrew.cmu.edu>, "Jared M. Oberhaus" <oberhaus+@CMU.EDU> writes:
> OK, I think I've figured out that the aete resource is the Dictionary
> for translating raw AppleEvent formats into plain English-like commands.
>
> But, where is the Template resource for ResEdit, so I can create and
> modify this resource?

I think I found a copy on the AppleScript developer CD. Trouble is, ResEdit
doesn't seem to be entirely reliable in its support for templates, and I've
hit problems trying to use it to open some aete resources.

Generally, if I want to know details of event IDs etc, I use MPW's DeRez to
decompile the aete into source form. If I don't, then the Script Editor's
"Open Dictionary" command suffices.

Lawrence D'Oliveiro                       fone: +64-7-856-2889
Info & Tech Services Division              fax: +64-7-838-4066
University of Waikato            electric mail: ldo@waikato.ac.nz
Hamilton, New Zealand    37^ 47' 26" S, 175^ 19' 7" E, GMT+12:00
My hard disk is a Mightgrosoft-free zone.

+++++++++++++++++++++++++++

>From knut.mork@admin.uio.no (Knut Mork)
Date: Wed, 30 Mar 1994 10:16:28 +0200
Organization: UniNett

In article <1994Mar30.111959.27070@waikato.ac.nz>, ldo@waikato.ac.nz
(Lawrence D'Oliveiro, Waikato University) wrote:

> In article <AhZyn_y00iV4E0fEpd@andrew.cmu.edu>, "Jared M. Oberhaus" <oberhaus+@CMU.EDU> writes:
> > OK, I think I've figured out that the aete resource is the Dictionary
> > for translating raw AppleEvent formats into plain English-like commands.
> >
> > But, where is the Template resource for ResEdit, so I can create and
> > modify this resource?
> 
> I think I found a copy on the AppleScript developer CD. Trouble is, ResEdit
> doesn't seem to be entirely reliable in its support for templates, and I've
> hit problems trying to use it to open some aete resources.
> 
> Generally, if I want to know details of event IDs etc, I use MPW's DeRez to
> decompile the aete into source form. If I don't, then the Script Editor's
> "Open Dictionary" command suffices.

The AppleScript developer CD also has a HyperCard stack for constructing
aete resources, which is extremely useful. It has Rez templates for MPW,
but those didn't work.. they compiled 'aete' resources which were invalid,
at least for me. Anybody else have this problem?

--Knut

+++++++++++++++++++++++++++

>From mlanett@netcom.com (Mark Lanett)
Date: Wed, 30 Mar 1994 08:14:12 GMT
Organization: Etch-a-Sketch Analysis and Design

ldo@waikato.ac.nz (Lawrence D'Oliveiro, Waikato University) writes:

>I think I found a copy on the AppleScript developer CD. Trouble is, ResEdit
>doesn't seem to be entirely reliable in its support for templates, and I've
>hit problems trying to use it to open some aete resources.

I don't have the AS toolkit, so when I felt like doing stuff with scripting
I had to figure out the format of the aete resource myself, and...
It looks like it can't be handled by ResEdit. Apparently all the "direct"
types like shorts and longs are word-aligned, but strings are not. There's no
way to create a template which handles word-alignment in ResEdit, alas, and
since the strings *aren't* aligned you can't make a template that works by
specifying ESTR instead of PSTR.

>...the Script Editor's
>"Open Dictionary" command suffices.

Never even thought of that one. Oh well, Sucks to be me.

>My hard disk is a Mightgrosoft-free zone.
I love MS products. *My* HD is a Symantec-free zone.
-- 
    Mark Lanett "...a bajillion brilliant Jobsian lithium licks"

+++++++++++++++++++++++++++

>From alldritt@mdd.comm.mot.com (Mark Alldritt)
Date: 30 Mar 1994 08:31:00 -0800
Organization: Motorola - Wireless Data Group; Richmond, BC

In <mlanettCnGxJo.Fov@netcom.com> mlanett@netcom.com (Mark Lanett) writes:

>ldo@waikato.ac.nz (Lawrence D'Oliveiro, Waikato University) writes:

>>I think I found a copy on the AppleScript developer CD. Trouble is, ResEdit
>>doesn't seem to be entirely reliable in its support for templates, and I've
>>hit problems trying to use it to open some aete resources.

>I don't have the AS toolkit, so when I felt like doing stuff with scripting
>I had to figure out the format of the aete resource myself, and...
>It looks like it can't be handled by ResEdit. Apparently all the "direct"
>types like shorts and longs are word-aligned, but strings are not. There's no
>way to create a template which handles word-alignment in ResEdit, alas, and
>since the strings *aren't* aligned you can't make a template that works by
>specifying ESTR instead of PSTR.

The AppleScript developer CD contains a TMPL resource for editing aete
resources which works reasonably well.  The problem is that TMPL layouts
can only handle a limited number of fields.  Application aete resources are
generally too large to be edited using the aete TMPL.

Another alternative is the aete editing Hypercard stack Apple provides on 
the AppleScript CD and (I think) on the Develop Bookmark CD.

However, if your just interested in know what events and classes a application
understands then the Script Editor's dictionary window is your best bet.

-Mark

+++++++++++++++++++++++++++

>From lai@apple.com (Ed Lai)
Date: 5 Apr 1994 17:04:19 GMT
Organization: Apple

In article <AhZyn_y00iV4E0fEpd@andrew.cmu.edu>, "Jared M. Oberhaus"
<oberhaus+@CMU.EDU> wrote:

> OK, I think I've figured out that the aete resource is the Dictionary
> for translating raw AppleEvent formats into plain English-like commands.
> 
> But, where is the Template resource for ResEdit, so I can create and
> modify this resource?
> 

It should be on the developer CD. You should also find a copy in
from ftp.apple.com in /pub/appleevents, although that may not be the
latest copy. The problem is that ResEdit template editing probably
uses the dialog manager and there can only be a limited number of
item in the dialog, any decent size aete would reach the limit so
for practical purpose it is useless for aete unless the aete is
are very simple like those in an OSAX.

You should be able to find the HyperCard aete editor stack in the 
same place.

> And if it doesn't exist yet, how do all you other guys do it? Not the
> hard way, I hope...
> 
> Help!!
> 
> ps-I saw the demo of Resorcerer and noticed that it supports aete
> resources, and looked around, and I supposed I could figure out how to
> do an aete resoure by looking at the hex offsets, etc., but I don't
> exactly feel like doing that!

I hear that Resorcerer is very good at editing aete resource, this is
certainly a solution for those who owns Resorcerer.

There is yet another solution. I have written an osax to translate
between resources and list using ResEdit template, and since aete is 
a resource with ResEdit template. In theory you can use the osax
and generate aete resource using AppleScript. I have not tried it
for a long time, but I suspect it is very slow so may not be
practical except for smaller aete. The osax can be found in
ftp.apple.com in /pub/lai/osax/pgmTool.sit.hqx

-- 
/* Disclaimer: All statments and opinions expressed are my own */
/* Edmund K. Lai                                               */
/* Apple Computer, MS303-3A                                    */
/* 20525 Mariani Ave,                                          */
/* Cupertino, CA 95014                                         */
/* (408)974-6272                                               */
zW@h9cOi

---------------------------

>From dsf5454@ritvax.isc.rit.edu
Subject: Help on recursive read of directory catalog..?
Date: Sun, 3 Apr 1994 16:59:38 GMT
Organization: Rochester Institute of Technology

Ok...basically, I have a crucial subroutine where I need to read a directory
tree below a specific subdirectory. However, I notice that I seem to be
crashing into the debugger with an error of #28 and somewhere around in a VBL
queue - stack and application heap collison. Therefore, I'm not sure how I
could efficiently code such a recursive call... suggestions or pointers,
anybody? I'm calling it with RecursiveDirList(1, 2L); in main to tell it to
start from root folder.

I have a newer version that was re-written, but the principle is the same...
Roughly speaking, here's what it looks like at the moment..:

-Dan
Internet:	dsf5454@ritvax.isc.rit.edu
BITNET/CREN:	dsf5454@ritvax.BITNET

long RecursiveDirList (t, dirIDToSearch)
short t;
long dirIDToSearch;
{
	short	index = 1;
	OSErr	err;
	long	curLine;

	do {

		myCPB.fileParam.ioFDirIndex = index;
		myCPB.fileParam.ioVRefNum = vRefNum;
		myCPB.fileParam.ioDirID = dirIDToSearch;
		myCPB.fileParam.ioNamePtr = (unsigned char *)fName;

		err = PBGetCatInfo ((CInfoPBPtr)&myCPB, 0);
		
		if (err == noErr) 
		{
			if (((myCPB.fileParam.ioFlAttrib >> 4) & 0x01) == 1) 
			{
				RecursiveDirList(t+1,myCPB.fileParam.ioDirID);
				err = 0;
				/* It's a directory; we want to go deeper into
				the subdirectories */
			} 
			else 
			{
				/* it's a file */
			}
		}
		index++;
	} while ((err != fnfErr) && (err == noErr));
}
                               

+++++++++++++++++++++++++++

>From allon@intercon.com (Allon Stern)
Date: Tue,  5 Apr 1994 03:09:18 -0400
Organization: InterCon Systems Corporation, Herndon VA.

In article <1994Apr3.165938.3952@ultb.isc.rit.edu>, 
dsf5454@ritvax.isc.rit.edu writes:
> 
> Ok...basically, I have a crucial subroutine where I need to read a 
> directory tree below a specific subdirectory. However, I notice that I 
> seem to be crashing into the debugger with an error of #28 and 
> somewhere around in a VBL queue - stack and application heap collison. 
> Therefore, I'm not sure how I could efficiently code such a recursive 
> call... suggestions or pointers, anybody? I'm calling it with 
> RecursiveDirList(1, 2L); in main to tell it to start from root folder. 
> 
> I have a newer version that was re-written, but the principle is the 
> same... 
> Roughly speaking, here's what it looks like at the moment..: 
> 
> -Dan 
> Internet:	dsf5454@ritvax.isc.rit.edu 
> BITNET/CREN:	dsf5454@ritvax.BITNET 

It seems to me that you could probably do this non-recusively.
Keep in mind that you will end up using stack space for each recursive call,
and also that PBGetCatInfo and the file manager may also use some stack
space.

Rather than using the stack to store where you were, you could build your own
stack structure, say, using an array.  This would minimize stack usage, and
would also store the same information much more efficiently.  If it is a
transient thing (i.e. you use a bunch of storage for a short time, then would
release it), you could use temporary memory for it.  You could also use a
dynamic array, and if you run out of room in your own heap, you could transfer
the contents to a handle in temporary memory space.

You could try increasing the memory requested for the program, but I think
the best thing in this case is to find a non-recursive way to do it.
--
Allon


| Allon Stern        | 703-709-5557 |  Ring around the Internet    |
| allon@intercon.com |    KE4FYL    |  A packet with a bit not set |
+--------------------+--------------+  ENQ ACK ENQ ACK             |
|  All opinions above are my own.   |  We all go down!             |
|------------------------------------------------------------------|



+++++++++++++++++++++++++++

>From d88-jwa@mumrik.nada.kth.se (Jon Wätte)
Date: 5 Apr 1994 08:13:09 GMT
Organization: The Royal Institute of Technology

In <9404050309.AA18685@hazmat.intercon.com> allon@intercon.com (Allon Stern) writes:

>> Ok...basically, I have a crucial subroutine where I need to read a 
>> directory tree below a specific subdirectory. However, I notice that I 
>> seem to be crashing into the debugger with an error of #28 and 
>> somewhere around in a VBL queue - stack and application heap collison. 

>It seems to me that you could probably do this non-recusively.
>Keep in mind that you will end up using stack space for each recursive call,
>and also that PBGetCatInfo and the file manager may also use some stack
>space.

Yeah, but PBGetCatInfo will give the stack space back when it's done.
(It should be called as PBGetCatInfoSync() by the way, or maybe
PBGetCatInfoAsync ( ) ; while ( rec . ioResult == 1 ) Yield ( ) ;)

Anyway, his routine only ate 4+4+4+4+4+4+4 bytes per recursion
(2 int arguments, a6 link, return address, three int locals, given
4-byte ints)
At 28 bytes per recursion, you need > 1000 folders deep structures
to hit any real problems, unless you have MASSIVE parameter blocks
and Str255s on the stack higher up. (Putting Str255s on the stack
is a BAD idea btw)

What he could do is call StackSpace() (or whatever) before each
recursive call, and call Debugger() if it goes below a K or so.
Most probably, however, he's having some other bug which is causing
the problem.
-- 
 -- Jon W{tte, h+@nada.kth.se, Mac Hacker Deluxe --

   What we need is a good GNU [...] licence manager implementation.
                     -- Raphael Manfredi

+++++++++++++++++++++++++++

>From jumplong@aol.com (Jump Long)
Date: 5 Apr 1994 12:44:02 -0400
Organization: America Online, Inc. (1-800-827-6364)

In article <1994Apr3.165938.3952@ultb.isc.rit.edu>, dsf5454@ritvax.isc.rit.edu
writes:

> Ok...basically, I have a crucial subroutine where I need to read a directory
> tree below a specific subdirectory. However, I notice that I seem to be
> crashing into the debugger with an error of #28 and somewhere around in a VBL
> queue - stack and application heap collison. Therefore, I'm not sure how I
> could efficiently code such a recursive call... suggestions or pointers,
> anybody? I'm calling it with RecursiveDirList(1, 2L); in main to tell it to
> start from root folder.

You're using too much stack in your recursive routine - either that, of you
don't have much stack when you start your recursive routine.

You can check the stack space before you start with the StackSpace function. 
If that isn't your problem, then you might want to look at one of the recursive
routines I put in DTS's MoreFiles File Manager sample code. In particular, the
routines in DirectoryCopy.c show how you can use very little stack space per
level in a recursive routine that enumerates a directory structure on a
Macintosh disk volume.

The code in MoreFiles is based on the ideas presented in the Technical Note "

+++++++++++++++++++++++++++

>From jumplong@aol.com (Jump Long)
Date: 5 Apr 1994 12:45:03 -0400
Organization: America Online, Inc. (1-800-827-6364)

In article <1994Apr3.165938.3952@ultb.isc.rit.edu>, dsf5454@ritvax.isc.rit.edu
writes:

Ooops, hit the send button too soon... starting where I left off.

on ideas presented in the Technical Note "FL 31 - Searching Volumes-Solutions
and Problems".

Hope that helps.

Jim Luther
Apple DTS

---------------------------

>From jbell@garnet.berkeley.edu (John E. Bell)
Subject: How can I display TIFF?
Date: 22 Mar 1994 11:07:24 GMT
Organization: University of California, Berkeley


How can I display a TIFF file in my program? Are there library routines
somewhere?

Thanks - John E. Bell

+++++++++++++++++++++++++++

>From mssmith@afterlife.ncsc.mil (M. Scott Smith)
Date: Tue, 22 Mar 1994 12:18:47 GMT
Organization: The Great Beyond

In article <2mmjhc$sno@agate.berkeley.edu> jbell@garnet.berkeley.edu (John E. Bell) writes:
>
>How can I display a TIFF file in my program? Are there library routines
>somewhere?
>
>Thanks - John E. Bell

John,

   There are some samples floating about that show how to read the TIFF
format.  The TIFF format is kind of an ugly thing to read; it takes a lot
of work.  It's designed to be a very flexible file format, so each TIFF
file starts out with a number of "tags" which can contain everything from
the expected width and height of the image, to the resolution, even comments
and tags/fields which are specific to certain applications.

   I guess in order to claim that your program can read TIFF, you need to
intelligently read a certain number of these tags, and you can ignore the
rest.

   The tags will give you information like offsets for finding the actual
data, etc.

   Really kind of yechy.  But, alas: if I remember correctly, NIH Image,
a program which you can find at the usual FTP sites, reads color TIFF
images.  The only problem is that it's written in Pascal.  (That may
not be a problem for some people.)  You can probably easily translate it
to C.

   I've written black and white TIFF-reading code (not color), and I
suppose I might some day spruce that up to serve as an example.  (Right now,
I think I'd be embarassed to claim credit for it.  It does read files
quickly though.)

   What I'd like is a nice C++ library that lets you save/read PixMaps
to/from a multitude of file formats, such as PICT, TIFF, PostScript,
and the like.  I don't think such a thing exists (anyone know??), so
it seems whenever you want to write a program with these capabilities,
you need to re-invent the wheel.

   Maybe people could get together and build this library; between us,
I'm sure we have all of the capability.

Anyway, good luck with the TIFF reading.  (Trivia: TIFF stands for
Tagged Image File Format.)

- Scott

- -
M. Scott Smith   (mssmith@afterlife.ncsc.mil || umsmith@mcs.drexel.edu)

  Mac developer. Student.  Not a member of the Michael Bolton fan club.


+++++++++++++++++++++++++++

>From perm@csd.uu.se (Per Mildner)
Date: 22 Mar 1994 16:04:36 GMT
Organization: Computing Science Dept.,Uppsala University, Sweden

I recommend looking for LIBTIFF with Archie, it reads and writes
various tiff formats.  My only problem with that is that I know an
application (Optix by Blueridge) that unpacks CCITT Group IV (fax)
very much faster than the seemingly optimized LIBTIFF code (on a
Quadra 700).
Regards,
--
Per Mildner			Per.Mildner@CSD.UU.SE
Computing Science Dept.		tel: +46 18 181049
Uppsala University, Sweden	fax: +46 18 511925


+++++++++++++++++++++++++++

>From markhanrek@aol.com (MarkHanrek)
Date: 22 Mar 1994 20:45:07 -0500
Organization: America Online, Inc. (1-800-827-6364)

Yes, there is some TIFF source code available from a variety of places.

The place to start, though, is the Aldus Developer Desk, which is on
CompuServe, from which you will be able to download the current TIFF Level 6.0
specification, example source code (for level 5), and some TIFF pictures which
are good to use for testing.

Or you could contact Aldus directly. Perhaps there is Internet access to them

Mark Hanrek


+++++++++++++++++++++++++++

>From kenlong@netcom.com (Ken Long)
Date: Wed, 23 Mar 1994 02:51:35 GMT
Organization: NETCOM On-line Communication Services (408 241-9760 guest)

There's some TIFF reader code on AOL.  Next time I log on I'll get a 
description of it for you.  Seems like I saw some elsewhere, too.  i'll 
check it out.

-Ken-

+++++++++++++++++++++++++++

>From xinwei@otter.Stanford.EDU (Sha Xin Wei)
Date: 5 Apr 1994 21:39:26 GMT
Organization: Stanford University

M. Scott Smith writes
> In article <2mmjhc$sno@agate.berkeley.edu> jbell@garnet.berkeley.edu (John E.  
Bell) writes:
> >
> >How can I display a TIFF file in my program? Are there library routines
> >somewhere?
> >
> >Thanks - John E. Bell
> 
> John,
> 
>    There are some samples floating about that show how to read the TIFF
> format.>    Maybe people could get together and build this library; between  
us,
> I'm sure we have all of the capability.
> 
> Anyway, good luck with the TIFF reading.  (Trivia: TIFF stands for
> Tagged Image File Format.)
> 
> - Scott
> 
> ---
> M. Scott Smith   (mssmith@afterlife.ncsc.mil || umsmith@mcs.drexel.edu)


By the way, in the unix world there's a whole wonderful library named pbmplus  
by Jef Pozkaner, written in C, which is in the public domain.  It provides a  
slew of tools for conversion between 12-40 different graphics formats  
(depending on how you count) -- PICT, TIFF, PS, GIF, etc., and much more: color  
histogram, diffusion, quatization, scaling, rotation, palette remaps, even  
general convolution given a user-supplied kernel.   All in public domain.    
PixelMagician, a NeXTSTEP application has extended pbmplus to handle pixmaps  
with alpha channel, and a few more formats.

If anyone has compiled this under MPW or ThinkC, could they please publish it?
Thanks.

Sha Xin Wei
Stanford University

---------------------------

>From Vik_Rubenfeld@lamg.com (Vik Rubenfeld)
Subject: Inverting a button in a dialog
Date: 05 Apr 1994 00:21:14 -0000
Organization: (none)

I've set up a filter procedure that returns the item number of the cancel
button if the user hits the escape key. This lets the user dismiss the dialog
from the keyboard, without using the mouse. It works great. However, it does
not invert the cancel button before dismissing the dialog.

So, could someone upload some sample code that highlights (inverses) a
standard dialog button? Thanks.

+++++++++++++++++++++++++++

>From t-gaul@i-link.com (Troy Gaul)
Date: Tue, 05 Apr 1994 10:48:58 -0500
Organization: I-Link, Ltd.

In article <101974014.48568985@lamgnet.lamg.com>, Vik_Rubenfeld@lamg.com
(Vik Rubenfeld) wrote:

> I've set up a filter procedure that returns the item number of the cancel
> button if the user hits the escape key. This lets the user dismiss the dialog
> from the keyboard, without using the mouse. It works great. However, it does
> not invert the cancel button before dismissing the dialog.
> 
> So, could someone upload some sample code that highlights (inverses) a
> standard dialog button? Thanks.

ControlHandle
GetControlItemHandle(DialogPtr dlog, short item) {
  short  iKind;
  Handle iHandle;
  Rect   iRect;

  GetDItem(dlog, item, &iKind, &iHandle, &iRect);
  switch (iKind & ~itemDisable) {
    case btnCtrl + ctrlItem:
    case chkCtrl + ctrlItem:
    case radCtrl + ctrlItem:
    case resCtrl + ctrlItem:
      return (ControlHandle) iHandle;
  }
  return nil;
}

void 
FlashDItem(DialogPtr dlog, short item) {
  long ignored;
  ControlHandle button;
	
  button = GetControlItemHandle(dlog, item);
  if (button != nil && (**button).contrlHilite != 255) {
    HiliteControl(button, 1);
    Delay(8, &ignored);          // the Apple prescribed period of time
    HiliteControl(button, 0);
  }
}

Of course, to flash the cancel button, you just call:

  FlashDItem(dlog, cancel);

_troy
//////// //////___Troy Gaul_________________________t-gaul@i-link.com__ //
  //    //       I-Link, Ltd. ; West Des Moines, Iowa                  //
 //    //  //   "Iungo ergo sum." (I-Link, therefore I am.)           //
//    //////________________________________________________________ //

+++++++++++++++++++++++++++

>From charlesworth@andyne.on.ca (Dave Charlesworth)
Date: Tue, 5 Apr 1994 22:15:45 GMT
Organization: Andyne Computing

I don't have sample code handy but you'll find it really simple to
write.  Here's the basic outline:
1.  Get the Cancel button item from the dialog.
2.  Coerce it to a control.
3.  Tell the control to highlight.
4.  Wait long enough for the user to see it.
5.  Go on with dismissing your dialog...

.../dave   Dave Charlesworth

---------------------------

>From RobTerrell@aol.com (Rob Terrell)
Subject: Looking for styled TE replacement?
Date: 4 Apr 1994 17:55:08 GMT
Organization: Jecta Development Corp.

A few weeks back someone mentioned a styled text edit replacement
called "WASTE". The poster claimed it would do styles as well as >32K
of text.

I've never heard of it, nor has archie. Any ideas?

Thanks,


Rob
Software Designs Unlimited



+++++++++++++++++++++++++++

>From mlanett@netcom.com (Mark Lanett)
Date: Tue, 5 Apr 1994 02:30:27 GMT
Organization: Etch-a-Sketch Analysis and Design

RobTerrell@aol.com (Rob Terrell) writes:

>A few weeks back someone mentioned a styled text edit replacement
>called "WASTE". The poster claimed it would do styles as well as >32K
>of text.

Host: ghost.dsi.unimi.it (149.132.1.2)
Path: pub2/papers/piovanel/
-- 
    Mark Lanett "...a bajillion brilliant Jobsian lithium licks"

---------------------------

>From rdm4@acpub.duke.edu (RYAN MARTELL)
Subject: Mounting AFPServer volume...
Date: 5 Apr 1994 21:26:42 GMT
Organization: Duke University, Durham, N.C.


Does anyone have any sample code showing how to mount a volume from an
AFPServer remotely?  I already have the address of the Server, and
figure I need to use AFPCommand, but I don't know how to send the
proper name/password verification data- or is there just a standard
call to do all of that for you? (Essentially, I am trying to write
something that acts as the chooser for you..)

Any help would be appreciated, please reply by email.

-Ryan Martell

+++++++++++++++++++++++++++

>From jumplong@aol.com (Jump Long)
Date: 5 Apr 1994 23:55:02 -0400
Organization: America Online, Inc. (1-800-827-6364)

In article <2nsl2i$p38@news.duke.edu>, rdm4@acpub.duke.edu (RYAN MARTELL)
writes:

> Does anyone have any sample code showing how to mount a volume from an
> AFPServer remotely?  I already have the address of the Server, and
> figure I need to use AFPCommand, but I don't know how to send the
> proper name/password verification data- or is there just a standard
> call to do all of that for you? (Essentially, I am trying to write
> something that acts as the chooser for you..)

There's two ways to do this so that the volume is mounted by the File Manager
(you probably don't want to mount the volume using AFP commands because the
File Manager won't know about the volume if mounted that way).

The first method is to use the Alias Manager.  Create a new alias record using
the NewAliasMinimalFromFullPath function.  Then, resolve the alias with the
ResolveAlias function.  The Alias Manager will take care of the user interface
for you.

The second way is to use the File Manager's PBVolumeMount function. You'll need
to build an AFP VolumeMountInfo record to pass to PBVolumeMount.  The
BuildAFPVolMountInfo function in the Apple DTS MoreFiles sample code will do
that for you.  MoreFiles also includes a high level versions of the VolumeMount
related calls if you don't like using parameter blocks.

- Jim Luther


---------------------------

>From Frank Price <wprice@jarthur.claremont.edu>
Subject: PPC ThreadManager w-CodeWarrior
Date: 5 Apr 1994 16:38:15 GMT
Organization: Pomona College

Has anybody managed to get the ThreadManager 2.0d7 release running native
on the PowerPC using CodeWarrior?  All I get is crashes on NewThread which
somewhat limits what one can do.  Other simple calls like GetCurrentThread
seem to work fine.  Haven't tried this with MPW yet, but I don't imagine it
would be any different (but if it wasn't different, how could they have
released this??)  Compiling the same code for 68k runs fine.  No, I'm not
trying to make preemptive threads, and I believe it would return an error
even if I had.  Also, did anyone notice that the Threads.h header file
doesn't even compile?  An excess comma has to be removed from inside an
enum.  I get the feeling this went out the door a little too quickly.

-Frank
_______________________________________________________________________
| Frank Price   |  wprice@jarthur.claremont.edu                        |
|_______________|______________________________________________________|

+++++++++++++++++++++++++++

>From REDDEN@applelink.apple.com (Kevin Redden)
Date: Tue, 5 Apr 1994 21:07:53 GMT
Organization: Apple Computer

In article <2ns45n$ht9@jaws.cs.hmc.edu>, Frank Price
<wprice@jarthur.claremont.edu> wrote:

> Has anybody managed to get the ThreadManager 2.0d7 release running native
> on the PowerPC using CodeWarrior?  All I get is crashes on NewThread which
> somewhat limits what one can do.  Other simple calls like GetCurrentThread
> seem to work fine.  Haven't tried this with MPW yet, but I don't imagine it
> would be any different (but if it wasn't different, how could they have
> released this??)  Compiling the same code for 68k runs fine.  No, I'm not
> trying to make preemptive threads, and I believe it would return an error
> even if I had.  Also, did anyone notice that the Threads.h header file
> doesn't even compile?  An excess comma has to be removed from inside an
> enum.  I get the feeling this went out the door a little too quickly.
> 

I had some code that I got working with it quite a while ago (I'm not
positive that it was 2.0d7 but I think so because I got the same comma
error), but I never tried it with CodeWarrier, just Apples SDK for PowerPC.

The comma compiles straight through with CFront and I believe that PPCC
only flags it when -strict on is specified.  I did bring the comma problem
up with one of the engineers that works on the Threads package but it was
already too late.  Hopefully in the next versionÉ

Kevin
Tech Lead, Macintosh On RISC SDK 2.0

+++++++++++++++++++++++++++

>From bpost@apple.com (Brad Post)
Date: Wed, 6 Apr 1994 01:44:05 GMT
Organization: Apple Computer

In article <2ns45n$ht9@jaws.cs.hmc.edu>, Frank Price
<wprice@jarthur.claremont.edu> wrote:
> 
> Has anybody managed to get the ThreadManager 2.0d7 release running native
> on the PowerPC using CodeWarrior?  All I get is crashes on NewThread which
> somewhat limits what one can do.  Other simple calls like GetCurrentThread
> seem to work fine.  Haven't tried this with MPW yet, but I don't imagine it
> would be any different (but if it wasn't different, how could they have
> released this??)  Compiling the same code for 68k runs fine.  No, I'm not
> trying to make preemptive threads, and I believe it would return an error
> even if I had.  Also, did anyone notice that the Threads.h header file
> doesn't even compile?  An excess comma has to be removed from inside an
> enum.  I get the feeling this went out the door a little too quickly.


Okay this is a simple problem to fix (crashing that is).  The problem is
that you are most likely using the Voodoo Monkey Debugger Init.  It doesn't
know about the native thread manager, so it's trying to do some gross stuff
that it shouldn't be doing.  Remove it and you should work just fine.  As
for the comma, it slipped out, and it should be fixed in the latest/next
release (or it's an exercise for the programmer !-)

Brad 'SMMFD' Post

+++++++++++++++++++++++++++

>From jwbaxter@olympus.net (John W. Baxter)
Date: Tue, 05 Apr 1994 17:46:04 -0700
Organization: Internet for the Olympic Peninsula

In article <2ns45n$ht9@jaws.cs.hmc.edu>, Frank Price
<wprice@jarthur.claremont.edu> wrote:

> Has anybody managed to get the ThreadManager 2.0d7 release running native
> on the PowerPC using CodeWarrior?  All I get is crashes on NewThread which
> somewhat limits what one can do.

You might have better luck with the ThreadManager 2.0 (no qualification)
which appeared on the April Developer CD.  I don't believe this one is the
last word in Thread Managers, either, though.

-- 
John Baxter    Port Ludlow, WA, USA  [West shore, Puget Sound]
   jwbaxter@pt.olympus.net

+++++++++++++++++++++++++++

>From Frank Price <wprice@jarthur.claremont.edu>
Date: 6 Apr 1994 17:04:27 GMT
Organization: Pomona College

In article <bpost-050494183834@bsp.apple.com> Brad Post, bpost@apple.com
writes:
>Okay this is a simple problem to fix (crashing that is).  The problem is
>that you are most likely using the Voodoo Monkey Debugger Init.  It doesn't
>know about the native thread manager, so it's trying to do some gross stuff
>that it shouldn't be doing.  Remove it and you should work just fine.  As
>for the comma, it slipped out, and it should be fixed in the latest/next
>release (or it's an exercise for the programmer !-)

I wasn't and haven't ever used Voodoo Monkey.  I also took the suggestions
of others to get the 2.0 release off the April CD instead of 2.0d7.  Same
problem.  Compiling the simple NativeThreads app linking with the xcoff
file, it always crashes with an "access fault exception" at NewThread. 
Would appreciate it if someone could try this out to confirm what I'm
saying.  I'm using DR2 of CodeWarrior.  And again, everything still works
fine for the same code in 68k on the same machine.

-Frank
_______________________________________________________________________
| Frank Price   |  wprice@jarthur.claremont.edu                        |
|_______________|______________________________________________________|

---------------------------

>From piper@char.vnet.net (John Wash)
Subject: RJW: Retrieving application name from OSType Creator
Date: 31 Mar 1994 17:04:19 -0500
Organization: Vnet Internet Access, Inc. - Charlotte, NC. (704) 374-0779


Hi.  This is done in the Finder, so I'm sure there's a simple way to do 
it.  I have a document reference stored in a database.  I know it's 
four-character Creator.  How do I resolve that Creator into a path to the 
application?

I've looked through my old Inside Macs.  It seems to me that it should be 
just a Toolbox call, but I can't seem to find it. 

E-mail or followup messages appreciated.

Thanks!

John Wash
piper@vnet.net

-- 
-
John Wash                                               piper@vnet.net
Wee Doggie Musical Instruments                          919-851-4620
"Bagpipes are the missing link between music and noise." --Gordon Mooney

+++++++++++++++++++++++++++

>From jumplong@aol.com (Jump Long)
Date: 5 Apr 1994 12:34:09 -0400
Organization: America Online, Inc. (1-800-827-6364)

In article <2nfhd3$1b3@char.vnet.net>, piper@char.vnet.net (John Wash) writes:

> How do I resolve that Creator into a path to the 
> application?

I'd start by using the Process Manager to look through the list of open
processes to see if the application is already running.

If the process isn't running, then use the Desktop Manager's PBDTGetAPPL
function on the mounted volumes to try to locate the application. You'll want
to start with the volume the file with the creator is on and if you don't find
a match there, go on to the boot volume, local volumes, and then last, remote
volumes.  So, until you find a match, here's the steps you'll need to take:

1) Make sure the volume supports the Desktop Manager using PBHGetVolParms
(check for the bHasDesktopMgr attribute bit).

2) Open the desktop database (i.e., get the dtRefNum needed to make calls to
the Desktop Manager) on that volume using PBDTOpenInform. If PBDTOpenInform
fails with a paramErr, then try again using PBDTGetPath to open the desktop
database.

3) Call PBDTGetAPPL using the creator of the application you want to find.
Start with ioIndex = 0 to get the newest match in the desktop database, check
to make sure the application match is really there and if not, increment the
index, call PBDTGetAPPL again to get the next match. If you get to the end of
the match list on the volume (with an afpItemNotFound error), try the next
volume.

OK, so what if you don't find a match using the Desktop Manager, then you'll
probably want to try any volumes that *don't* support the Desktop Manager (and
thus, have the old Desktop resource file). There's more than one way to do
this. You could simply search the volume (using PBCatSearch) for a file with
the creator and a file type of 'APPL'.  Or, if you really wanted, you could use
the Desktop file on the disk volume. Since the Finder already has the Desktop
file open and you won't be able to open it with the Resource Manager, you'll
need to make a temporary copy of the Desktop file. You can use the FileCopy
function in DTS's MoreFiles File Manager sample code to make the copy of the
Desktop file.  Then you can open it up with the Resource Manager and use the
'APPL' resourced to attempt to find a matching application.  The Technical Note
"Resources Contained in the Desktop File" describes the resources you'll find
in the Desktop file and in particular, the 'APPL' resources.

So, what else can you do if none of these things finds an application match?

You could search each mounted volume for for a file with the creator and a file
type of 'APPL'.  The CreatorTypeFileSearch function in the MoreFiles sample
uses CatSearch to quickly search a volume for a file specified by creator and
file type.

About the only thing you cannot do that the Finder does is look through the
Finder's open windows for a match.

- Jim Luther


---------------------------

>From stk@uropax.contrib.de (Stefan Kurth)
Subject: Range of OSErr's for private use?
Date: 5 Apr 1994 20:47:50 +0200
Organization: Contributed Software GbR

Almost every function that I write returns a value of type OSErr.
Sometimes it would be desirable to return a private error code of my
own, rather than one of the Apple ones. Is there a range of values
that I can safely use for my own #defines, without having to worry
whether Apple might use them for their system 9.0 implementation of
the espresso manager or something?

-Stefan

_____________________________________________________________________
Stefan Kurth              Berlin, Germany              stk@contrib.de

+++++++++++++++++++++++++++

>From gurgle@netcom.com (Pete Gontier)
Date: Wed, 6 Apr 1994 19:00:25 GMT
Organization: cellular

stk@uropax.contrib.de (Stefan Kurth) writes:

>Almost every function that I write returns a value of type OSErr.
>Sometimes it would be desirable to return a private error code of my
>own, rather than one of the Apple ones. Is there a range of values
>that I can safely use for my own #defines, without having to worry
>whether Apple might use them for their system 9.0 implementation of the
>espresso manager or something?

I was just thinking about this the other night, and my conclusion
was both yes and no.

It seems that the majority of error codes that Apple adds are negative.
They still have thousands left over, but they like to leave gaps in
between groups of error codes, so they might use positive codes in the
not-too-distant future. In fact, they already do use some of the lower
positive codes (for CTB calls). You could risk using relatively high
positive codes, but it probably wouldn't work in the long run and it
wouldn't even be guranateed to work in the short run.

One alternative I thought of is to simply return a 'long'. Since an
'OSErr' is nothing but a signed 16-bit quantity, you could use signed
values which don't fit in 16 bits for your error codes and return a
'long'. When you go to report the error, you could determine if the
value would have fit into an 'OSErr', and if it would have, that's what
it is, but if it wouldn't have, it must be one of your custom error
codes. The only problem with this is that if some of your routines
return an 'OSErr' and others return a 'CustomErr', C won't help you if
you forget and assign a 'CustomErr' to an 'OSErr'.

     (A certain Swede who shall be nameless is likely to pop in here and
     complain that not only should I really be advocating exceptions
     instead of error codes but also that Pascal would be much better
     for this scheme because it wouldn't allow you to screw up assigning
     the function result. To which I say: yeah, and your mother wears
     army boots! :-)

Of course, this is not what I personally do. Each of my modules has a
unique 8-bit "location code", and each module's functions which can
produce errors produce a 16-bit struct (fits nicely in a register)
which also contains a non-unique error code. Put the two codes together
and you have a description of which module caused the error and a a
description of what the error was which does not rely on 'OSErr'.
-- 
 Pete Gontier, CTO, Integer Poet Software; gurgle@netcom.com

---------------------------

>From guyyalif@tucson.Princeton.EDU (Guy Udassin Yalif)
Subject: Simple Q: Assigning char * to char []. How?
Date: Mon, 4 Apr 1994 21:27:46 GMT
Organization: Princeton University

This seems ridiculous: You should be able to assign a char * to 
an array of characters.  If you declare char a[10], a is a pointer
to the first of a contiguous group of 10 characters.  Yet when I try
this on THINK C 6.0.1, it tells me I cannot assign to an array.  If
I try to get around this with an assignment to &a[0], it tells
me that an lvalue is required.  This is an error I have gotten 
many times, and I do not know how to get around it.  ANY help would
be greatly appreciated!! Thanks for the time.  Guy
guyyalif@phoenix.princeton.edu

+++++++++++++++++++++++++++

>From zstern@adobe.com (Zalman Stern)
Date: Tue, 5 Apr 1994 06:32:24 GMT
Organization: Adobe Systems Incorporated

Guy Udassin Yalif writes
> This seems ridiculous: You should be able to assign a char * to 
> an array of characters.  If you declare char a[10], a is a pointer
> to the first of a contiguous group of 10 characters.

No, a represents an absolute address of ten characters of storage. There is  
no storage allocated for a pointer (e.g. 4 bytes holding an address) which  
points to the beginning of that ten byte array. Another way to put this is  
that the declarations "extern char a[]" and "extern char *a" are very  
different. The best way to see this is to look at the assembly code  
generated for the following function:

extern char an_array[];
extern char *a_pointer;

int foo(void)
{
	int return_value;

	return_value = an_array[42];
	return_value += a_pointer[42];

	return return_value;
}

> Yet when I try
> this on THINK C 6.0.1, it tells me I cannot assign to an array.  If
> I try to get around this with an assignment to &a[0], it tells
> me that an lvalue is required.  This is an error I have gotten 
> many times, and I do not know how to get around it.  ANY help would
> be greatly appreciated!! Thanks for the time.  Guy

The C language specification says that array values are automatically  
converted to pointer values in most circumstances. It does not say they are  
the same thing. Think-C is trying to tell you that you do not understand the  
language. A good reading of the ANSI-C standard or other rigorous C  
documentation might be in order.
--
Zalman Stern		   zalman@adobe.com		    (415) 962 3824
Adobe Systems, 1585 Charleston Rd., POB 7900, Mountain View, CA 94039-7900
"Do right, and risk consequences." Motto of Sam Houston (via Molly Ivins)

+++++++++++++++++++++++++++

>From d88-jwa@mumrik.nada.kth.se (Jon Wätte)
Date: 5 Apr 1994 08:00:58 GMT
Organization: The Royal Institute of Technology

In <1994Apr4.212746.24171@Princeton.EDU> guyyalif@tucson.Princeton.EDU (Guy Udassin Yalif) writes:

>This seems ridiculous: You should be able to assign a char * to 
>an array of characters.  If you declare char a[10], a is a pointer

No, you should not. It's right there in K&R 2nd ed (and, indeed,
1st ed as well)

That's because the array is a CONSTANT pointer to a slab of memory
that lives on the stack or in global space. You can't just move that
slab of memory; all you can do is copy stuff into or out of the
memory the array is "pointing to".

>me that an lvalue is required.  This is an error I have gotten 
>many times, and I do not know how to get around it.  ANY help would

You can't. If you want to shove stuff into the actual array memory,
you should use strcpy() or memcpy() or possibly BlockMove(). If you
want a variable that can point at different arrays, you should use
a pointer variable and initialize it first.
-- 
 -- Jon W{tte, h+@nada.kth.se, Mac Hacker Deluxe --

   What we need is a good GNU [...] licence manager implementation.
                     -- Raphael Manfredi

---------------------------

>From potterf@tartarus.uwa.edu.au (Felix Potter)
Subject: [Q] Validity of a memory address
Date: 28 Mar 1994 06:54:22 GMT
Organization: The University of Western Australia

How does one check that a pointer is valid?  (Without dereferencing it and
possibly crashing the machine.)  At the moment my definition for a pointer
that is at least _possibly_ OK is:
    not null (ie. not equal to 0), and
    even (ie. not divisble by 2), and
    > ApplZone (low memory global), and
    < ApplLimit (low memory global)
So if my pointer fails one of these conditions, then I know straight away
that something has gone wrong.  Does anyone have any comments on either how
reasonable or, on the other hand, how totally stupid the above description is?
Any suggestions for better ways of going about it?

The reason I want to be able to check _quickly_ whether a pointer is valid
or not is that I am writing a few little 'wrapper' macros/functions to
go around the standard C malloc/calloc/realloc/free functions.  I realise that
just about every man and his dog has done this before me, but hey, I'm
having fun :)  One of the things I would like to be able to do is detect
heap corruption, and corruption of my record keeping information in
particular, and so I'd like to be able to tell if dereferencing a pointer
is going to crash my code or not before I do it, in a few strategic places.
Hence my above query.

Also, I'm going to be asking in come more general-purpose newsgroups, but
does anyone have any ideas about how to check pointers on other systems,
DOS and UNIX in particular?

Thanks for any help (or even idle flames, if I'm being particularly stupid :)

--
____________________________________________________________________________
 Felix Potter                                   potterf@lethe.uwa.edu.au
                                                potterf@tartarus.uwa.edu.au
 University of Western Australia, Perth.        potter_f@cs.uwa.edu.au

+++++++++++++++++++++++++++

>From potterf@tartarus.uwa.edu.au (Felix Potter)
Date: 28 Mar 1994 06:58:21 GMT
Organization: The University of Western Australia

I (potterf@tartarus.uwa.edu.au), in a fit of stupidity, wrote:
> How does one check that a pointer is valid?  (Without dereferencing it and
> possibly crashing the machine.)  At the moment my definition for a pointer
> that is at least _possibly_ OK is:
>     not null (ie. not equal to 0), and
>     even (ie. not divisble by 2), and
      ^^^^      ^^^^^^^^^^^^^^^^^

Oops *blush* :)

Obviously I mean "divisible by 2."

--
____________________________________________________________________________
 Felix Potter                                   potterf@lethe.uwa.edu.au
                                                potterf@tartarus.uwa.edu.au
 University of Western Australia, Perth.        potter_f@cs.uwa.edu.au

+++++++++++++++++++++++++++

>From Cameron Esfahani <dirty@guest.apple.com>
Date: Mon, 4 Apr 1994 04:42:50 GMT
Organization: Apple Computer, Inc.

In article <2n5uuu$ekq@styx.uwa.edu.au> Felix Potter,
potterf@tartarus.uwa.edu.au writes:
> How does one check that a pointer is valid?  (Without dereferencing it and
> possibly crashing the machine.)  At the moment my definition for a pointer
> that is at least _possibly_ OK is:
>     not null (ie. not equal to 0), and
>     even (ie. not divisble by 2), and
>     > ApplZone (low memory global), and
>     < ApplLimit (low memory global)
> So if my pointer fails one of these conditions, then I know straight away
> that something has gone wrong.  Does anyone have any comments on either how
> reasonable or, on the other hand, how totally stupid the above description
is?
> Any suggestions for better ways of going about it?
> 
> The reason I want to be able to check _quickly_ whether a pointer is valid
> or not is that I am writing a few little 'wrapper' macros/functions to
> go around the standard C malloc/calloc/realloc/free functions.  I realise
that
> just about every man and his dog has done this before me, but hey, I'm
> having fun :)  One of the things I would like to be able to do is detect
> heap corruption, and corruption of my record keeping information in
> particular, and so I'd like to be able to tell if dereferencing a pointer
> is going to crash my code or not before I do it, in a few strategic places.
> Hence my above query.
> 
> Also, I'm going to be asking in come more general-purpose newsgroups, but
> does anyone have any ideas about how to check pointers on other systems,
> DOS and UNIX in particular?
> 
> Thanks for any help (or even idle flames, if I'm being particularly stupid :)

Well, try installing your own Bus Error Handler and dereference
the pointer.  If you get called in your Bus Error Handler, then it
is not a valid pointer.  If you don't, it's valid....

Cameron Esfahani
dirty@apple.com

+++++++++++++++++++++++++++

>From ari@world.std.com (Ari I Halberstadt)
Date: Mon, 4 Apr 1994 06:26:22 GMT
Organization: The World Public Access UNIX, Brookline, MA

In article <2n5uuu$ekq@styx.uwa.edu.au>,
Felix Potter <potterf@tartarus.uwa.edu.au> wrote:
>How does one check that a pointer is valid?  (Without dereferencing it and
>possibly crashing the machine.)  At the moment my definition for a pointer
>that is at least _possibly_ OK is:
>    not null (ie. not equal to 0), and
>    even (ie. not divisble by 2), and
>    > ApplZone (low memory global), and
>    < ApplLimit (low memory global)
>So if my pointer fails one of these conditions, then I know straight away
>that something has gone wrong.  Does anyone have any comments on either how
>reasonable or, on the other hand, how totally stupid the above description is?
>Any suggestions for better ways of going about it?

Those are most of the tests you'd ever need. If you get a crash from
dereferencing a pointer after doing those checks then I'd be pretty
surprised. If you didn't mind the speed hit, you could look for the
pointer in some private table of valid pointers, using some sort of
binary tree, for instance. What these tests don't tell you, however,
is whether the contents of the pointer are valid.

What I've done in my applications is create a header that is placed at
the start of every pointer allocated. The header contains the size of
the pointer and a 4-byte signature code. A unique signature code is
used for every type of object I allocate. Then I do something like
this:

ObjectPtr NewObject(...) {
  ObjectPtr object = NewObjectPtr(sizeof(ObjectType), 'objp');
  ...
  assert(ValidObject(object));
  return(object);
}

Boolean ValidObject(ObjectPtr object) {
  if (! ValidObjectPtr(object, sizeof(ObjectType), 'objp')) return(false);
  ... more tests to check relationships among members of the object ...
}

void DoSomethingToObject(ObjectPtr object) {
  assert(ValidObject(object));
  ... use object ...
  assert(ValidObject(object));
}

The function ValidObjectPtr compares the size and ID code passed to it
as parameters with the values stored in the pointer's header. If they
differ, then the pointer is deemed invalid and false is returned. The
function also checks that the pointer's header and the end of the
pointer are within the application's heap zone.

As a further enhancement, you could store a trailer at the end of each
pointer. The trailer would contain the same 4-byte id code, and would
help catch errors that resulted from overwriting the end of the
pointer.

Also, when you allocate a pointer, fill it with some random (or not so
random) garbage. Something like 0xff works pretty well and is easy to
spot with a debugger. Using an odd value like 0xff is also good since
any members of the object that are pointers will be detected as
illegal pointers (since they'll be odd integers). When you dispose of
the pointer, again fill it with 0xff, since this will ensure that you
won't be able to use its contents without immediate errors.
Alternatively, you could zero the pointer when you allocate it, thus
removing the need to initialize members to a default value. However,
you should still fill the pointer with invalid data when it is
disposed of. My personal preference is to zero program objects but to
fill data pointers with 0xff bits. I disable the filling of memory
with 0xff bits for greater efficiency in the final application

If you were really paranoid, you could checksum the contents of the
pointer. For instance, instead of just calling ValidObject, you could
do something like:

  CheckInObject(object);
  ...
  CheckOutObject(object);

CheckInObject would calculate a checksum for the object and compare it
with the checksum saved in the pointer's header. If the checksums
differ, CheckInObject would fail (e.g., raise an exception). Then, it
would call ValidObject, and finally it would increment a counter to
indicate that the object had been checksummed and is in use. A value
greater than zero for the counter would prevent further calls to
CheckInObject from calculating a checksum or calling ValidObject.
CheckOutObject would decrement the counter each time it is called.
When the matching call to CheckOutObject is made (the counter is zero)
then a new checksum would be calculated for the object and the
checksum would be stored in the pointer's header. This should be
reasonably efficient and quite robust. You could make CheckInObject
and CheckOutObject macros that you could undefine when debug code was
disabled, or could change the macros so that they didn't do the
checksum operation in the final product. I haven't used this method
yet in my own applications, but I'm thinking of adding it.

>
>The reason I want to be able to check _quickly_ whether a pointer is valid
>or not is that I am writing a few little 'wrapper' macros/functions to
>go around the standard C malloc/calloc/realloc/free functions.  I realise that
>just about every man and his dog has done this before me, but hey, I'm
>having fun :)  One of the things I would like to be able to do is detect
>heap corruption, and corruption of my record keeping information in
>particular, and so I'd like to be able to tell if dereferencing a pointer
>is going to crash my code or not before I do it, in a few strategic places.
>Hence my above query.

To check the heap, you can just use MacsBug or TMON with heap check
enabled.  It works pretty well for most things. With TMON, one problem
is that by the time the heap is corrupted you can't figure out where
it failed in your code since all of the function labels are gone (it's
kind of a pain to get around this problem).

I like to validate memory on several different levels. The most
secure, but slowest, level is used during development. Less checks are
done in a beta level, and most (or all) checks are disabled in a
release version. However, lately, my memory checks have been fairly
fast (they used to be really slow), so I'm thinking of leaving at
least some of them in the release level.

Just checking the validity of pointers before calling memory
allocation and disposal routines is better than nothing, but overall
is not very secure. A pointer can easily become corrupted long before
free is ever called on it. You can also do a lot of extra verification
if you provide type information (like a four-byte code) in your code.
If you get fancy, you can even check the logical relationships among
the members of your objects. Checking logical relationships helps
catch additional programmatic errors.

>Also, I'm going to be asking in come more general-purpose newsgroups, but
>does anyone have any ideas about how to check pointers on other systems,
>DOS and UNIX in particular?

All of the checks I described above can be used on Unix systems. You
might have trouble finding something equivalent to ApplZone/ApplLimit
on Unix, but there's probably some equivalent, even if it's not
totally portable. The checks should probably also work with DOS
systems.  The interpretation of multiple byte character constants is
not defined by the language, so things like 'objp' may have to be
modified (e.g., use defines in some common memory header, don't
scatter the constants all over the place).

-- 
Ari Halberstadt    ari@world.std.com     #include <std/disclaimer.h>
"These beetles were long considered to be very rare because very few
entomologists look for beetles in the mountains, in winter, at night,
during snow storms." -- Purves W. K., et al, "Life: The Science of

+++++++++++++++++++++++++++

>From Alexander M. Rosenberg <alexr@apple.com>
Date: Mon, 4 Apr 1994 12:39:22 GMT
Organization: Hackers Anonymous

In article <1994Apr4.044250.27756@gallant.apple.com> Cameron Esfahani,
dirty@guest.apple.com writes:
> Well, try installing your own Bus Error Handler and dereference
> the pointer.  If you get called in your Bus Error Handler, then it
> is not a valid pointer.  If you don't, it's valid....

Trouble-maker.
- -------------------------------------------------------------------------
-  Alexander M. Rosenberg  - INTERNET: alexr@apple.com      - Yoyodyne    -
-  330 Waverley St., Apt B - UUCP:ucbvax!apple!alexr        - Propulsion  -
-  Palo Alto, CA 94301     -                                - Systems     -
-  (415) 329-8463          - Nobody is my employer so       - :-)         -
-                          - nobody cares what I say.       -             -

+++++++++++++++++++++++++++

>From jbrowne@zaphod.ncsa.uiuc.edu (Jim Browne)
Date: 5 Apr 94 05:34:57 GMT
Organization: University of Illinois at Urbana

ari@world.std.com (Ari I Halberstadt) writes:

>What I've done in my applications... 

Ari offers some good suggestions.  I once saw most of his suggestions
implemented in a library that was published in MacTech (MacTutor?) a while
back.  I can't remember the name of the article, but I remember the library
functioned on the principle of wrapping accesses to your data with MMUse and
MMUnuse.  The idea being that you had to call MMUse to be able to use your
data and must call MMUnuse as soon as you were done with it at that point.
This allowed for all kinds of nifty things the library could do behind your
back, like checksumming, checking for leaks, and even some primitive virtual
memory techniques.  If I could remember the names of the authors, I'd cite
them for you.

(Was it 'MemMan'?)
-- 
Jim Browne                                             | jbrowne@ncsa.uiuc.edu |
Head NCSA Mac Telnet Hacker, SDG System Administrator  | (217) 244-7798        |
<a href="http://www.ncsa.uiuc.edu/SDG/People/jbrowne/jbrowne.html">Click me</a>
                    "Not me, not yet, not that bad."

+++++++++++++++++++++++++++

>From jbrowne@zaphod.ncsa.uiuc.edu (Jim Browne)
Date: 5 Apr 94 07:28:00 GMT
Organization: University of Illinois at Urbana

jbrowne@zaphod.ncsa.uiuc.edu (Jim Browne) writes:

>ari@world.std.com (Ari I Halberstadt) writes:

>>What I've done in my applications... 

>Ari offers some good suggestions.  I once saw most of his suggestions
>implemented in a library that was published in MacTech (MacTutor?) a while
>back.  I can't remember the name of the article, but I remember the library
>functioned on the principle of wrapping accesses to your data with MMUse and
>MMUnuse.  The idea being that you had to call MMUse to be able to use your
>data and must call MMUnuse as soon as you were done with it at that point.
>This allowed for all kinds of nifty things the library could do behind your
>back, like checksumming, checking for leaks, and even some primitive virtual
>memory techniques.  If I could remember the names of the authors, I'd cite
>them for you.

Well, I've gone and looked up the article.  Thanks to those great people at
MacTech, I was able to grep the text index of articles for MacTech vols. 1-8.

Here's the reference:

"A Memory Manager for the Rest of US." by Jordan Zimmerman.  Vol. 7 No. 9 Pg. 40


-- 
Jim Browne                                             | jbrowne@ncsa.uiuc.edu |
Head NCSA Mac Telnet Hacker, SDG System Administrator  | (217) 244-7798        |
<a href="http://www.ncsa.uiuc.edu/SDG/People/jbrowne/jbrowne.html">Click me</a>
--"It's really weird... kind of like how Telnet is programmed." - Q. Koziol --

---------------------------

>From Ben J Fry <bf2c+@andrew.cmu.edu>
Subject: textedit bounds
Date: Tue,  5 Apr 1994 21:15:13 -0400
Organization: Freshman, Design, Carnegie Mellon, Pittsburgh, PA

does anyone know how to figure out the real boundaries for a styled
textedit field? that is, the natural boundary rectangle that the text
would fit in...

any help would be really appreciated.. i've nearly worn out my think
reference in trying to look for this... (if that's possible...)


ben

+++++++++++++++++++++++++++

>From ari@world.std.com (Ari I Halberstadt)
Date: Wed, 6 Apr 1994 21:27:41 GMT
Organization: The World Public Access UNIX, Brookline, MA

In article <MhcUoV200iV8IC9nE_@andrew.cmu.edu>,
Ben J Fry  <bf2c+@andrew.cmu.edu> wrote:
>does anyone know how to figure out the real boundaries for a styled
>textedit field? that is, the natural boundary rectangle that the text
>would fit in...
>
>any help would be really appreciated.. i've nearly worn out my think
>reference in trying to look for this... (if that's possible...)
>
>
>ben


Assuming you have word wrap turned on, you can find the width of the
text from the width of the destination rectangle. You can get the
height of the text with TEGetHeight. If you don't have word wrap
turned on, then you can either assume some maximum width, say 72
inches (5184 pixels). This is a large enough value to ensure that the
user can view most text, without making the horizontal scroll bar
as useless as a value of 32767 would. If you really want to be
accurate, then you you can calculate the width of the widest line in
the text. Here's some code to do this for both styled and unstyled
text. It knows way too much about TextEdit internals, but hey, it
works :-)

A few utility functions:

typedef unsigned long TicksType;
typedef SignedByte HandleStateType;

HandleStateType HandleLock(void *h)
{
  HandleStateType state;

  state = HGetState(h);
  HLock(h);
  return(state);
}

void HandleStateSet(void *h, HandleStateType state)
{
  HSetState(h, state);
}

// word wrap macros
#define TEWrap(te)          ((**(te)).crOnly >= 0)
#define TEWrapSet(wrap, te) ((void) ((**(te)).crOnly = ((wrap) ? 1 : -1)))

static Boolean TEOldStyle(TEHandle te)
{
	return((**te).txSize != -1);
}

/* Calculate the width of the widest line in the line range. If the
   operation takes longer than 'timeout' ticks to compute, then the
   last line reached is returned in 'finalLine', and the result of
   the function is the width of the widest line between 'firstLine'
   and 'finalLine'. Otherwise (i.e., no timeout), the result of the
   function is the width of the widest line and 'finalLine' is equal
   to 'lastLine'.
   
   The time out is provided since calculating the widths of the lines--
   especially for long styled text documents--can take quite a while, and
   doing so every time the document changes could be prohibitively slow.
   Instead, if 'finalLine' is less than 'lastLine', then after processing some
   events, the application can continue to call this function, passing
   the value of 'finalLine' as the 'firstLine' parameter, until 'finalLine'
   is equal to 'lastLine', at which point the maximum of the values returned
   by the successive calls to this function will be the width of the widest
   line in the original line range. */
static short TEWidthUnstyled(short firstLine, short lastLine,
   TicksType timeout, long *finalLine, TEHandle te)
{
   HandleStateType   svTextHandleState;    /* saved state of handle to text */
   GrafPtr           svPort;               /* saved graf port */
   Ptr               textPtr;              /* pointer to text */
   short             textWidth;            /* width of text */
   short             lineIndex;            /* current line number */
   short             lineLongest;          /* length of longest line */
   short             lineStart;            /* first character of line */
   short             lineEnd;              /* last character of line */
   long              startTicks            /* time when we started */
   short             txFont, txSize, txFace;   /* saved text state */
      
   require(TEOldStyle(te));
   
   /* quick out */
   if ((**te).nLines == 0) return(0);
      
   /* keep track of elapsed time */
   startTicks = TickCount();
   
   /* find longest line */
   lineStart = lineEnd = lineLongest = 0;
   for (lineIndex = firstLine; lineIndex < lastLine; lineIndex++) {

      /* remember length of longest line encountered so far */
      if (lineLongest < (**te).lineStarts[lineIndex + 1] - (**te).lineStarts[lineIndex]) {
         lineStart = (**te).lineStarts[lineIndex];
         lineEnd = (**te).lineStarts[lineIndex + 1];
         lineLongest = lineEnd - lineStart;
      }

      /* stop if took too long */
      if (TickCount() - startTicks > timeout)
         break;
   }
   
   /* calculate width of longest line */
   GetPort(&svPort);
   SetPort((**te).inPort);
   txFont = (**te).inPort->txFont;
   txSize = (**te).inPort->txSize;
   txFace = (**te).inPort->txFace;
   TextFont((**te).txFont);
   TextSize((**te).txSize);
   TextFace((**te).txFace);
   svTextHandleState = HandleLock((**te).hText);
   textPtr = *(**te).hText;
   textWidth = TextWidth(textPtr, lineStart, lineEnd - lineStart);
   HandleStateSet((**te).hText, svTextHandleState);
   TextFont(txFont);
   TextSize(txSize);
   TextFace(txFace);
   SetPort(svPort);

   *finalLine = lineIndex;
   return(textWidth);
}

/* see description of TEWidthUnstyled */
static short TEWidthStyled(short firstLine, short lastLine, TicksType timeout,
   long *finalLine, TEHandle te)
{
   HandleStateType   svTextHandleState;      /* saved state of handle to text */
   HandleStateType   svStyleHandleState;     /* saved state of handle to style record */
   HandleStateType   svStyleTableState;      /* saved state of handle to style table */
   GrafPtr           svPort;                 /* saved graf port */
   TEStylePtr        stylePtr;               /* text's style record */
   STPtr             styleTable;             /* text's style table */
   StyleRun         *runPtr;                 /* style runs array */
   short             runStart;               /* start of next style run */
   Ptr               textPtr;                /* pointer to text */
   short             textLength;             /* length of text run */
   short             textWidth;              /* width of text */
   short             lineWidth;              /* width of current line */
   short             lineIndex;              /* current line number */
   short             lineStart;              /* first character of line */
   short             lineEnd;                /* last character of line */
   long              startTicks;             /* time when we started */
   short             txFont, txSize, txFace; /* saved text state */
   
   require(! TEOldStyle(te));
   
   /* quick out */
   if ((**te).nLines == 0) return(0);

   /* keep track of elapsed time */
   startTicks = TickCount();
   
   /* setup port */   
   GetPort(&svPort);
   SetPort((**te).inPort);
   txFont = (**te).inPort->txFont;
   txSize = (**te).inPort->txSize;
   txFace = (**te).inPort->txFace;
   
   /* lock and dereference handles */
   svTextHandleState = HandleLock((**te).hText);
   textPtr = *(**te).hText;
   svStyleHandleState = HandleLock(GetStylHandle(te));
   stylePtr = *GetStylHandle(te);
   svStyleTableState = HandleLock(stylePtr->styleTab);
   styleTable = *stylePtr->styleTab;
   
   /* initialize loop */
   textWidth = 0;
   runPtr = stylePtr->runs;
   runStart = runPtr[1].startChar;
   TextFont(styleTable[runPtr->styleIndex].stFont);
   TextFace(styleTable[runPtr->styleIndex].stFace);
   TextSize(styleTable[runPtr->styleIndex].stSize);
   
   /* calculate widths of all lines */
   for (lineIndex = firstLine; lineIndex < lastLine; lineIndex++) {
   
      /* calculate width of line */
      lineWidth = 0;
      lineStart = (**te).lineStarts[lineIndex];
      lineEnd = (**te).lineStarts[lineIndex + 1];
      while (lineStart < lineEnd) {
         if (lineStart == runStart) {
            /* use next style run (assumes run array is sorted by startChar) */
            runPtr++;
            runStart = runPtr[1].startChar;
            check(runPtr->startChar == lineStart);
            TextFont(styleTable[runPtr->styleIndex].stFont);
            TextFace(styleTable[runPtr->styleIndex].stFace);
            TextSize(styleTable[runPtr->styleIndex].stSize);
         }
         /* calculate width of style run */
         check(lineStart < runStart);
         textLength = min(lineEnd - lineStart, runStart - lineStart);
         check(textLength >= 1);
         lineWidth += TextWidth(textPtr, lineStart, textLength);
         lineStart += textLength;
      }
      textWidth = max(textWidth, lineWidth);
      
      /* stop if took too long */
      if (TickCount() - startTicks > timeout)
         break;
   }

   /* restore saved state */
   HandleStateSet(RecoverHandle((Ptr) styleTable), svStyleTableState);
   HandleStateSet(RecoverHandle((Ptr) stylePtr), svStyleHandleState);
   HandleStateSet(RecoverHandle((Ptr) textPtr), svTextHandleState);
   TextFont(txFont);
   TextSize(txSize);
   TextFace(txFace);
   SetPort(svPort);
   
   *finalLine = lineIndex;
   return(textWidth);
}

/* see description of TEWidthUnstyled */
short TEWidthTimeout(short start, short end,
   TicksType timeout, long *final, TEHandle te)
{
   return(TEOldStyle(te) ? TEWidthUnstyled(start, end, timeout, final, te) :
                           TEWidthStyled(start, end, timeout, final, te));
}

/* calculate the width of the widest line in the line range */
short TEWidth(short start, short end, TEHandle te)
{
   return(TEWidthTimeout(start, end, LONG_MAX, 0, te));
}
-- 
Ari Halberstadt    ari@world.std.com     #include <std/disclaimer.h>
"These beetles were long considered to be very rare because very few
entomologists look for beetles in the mountains, in winter, at night,
during snow storms." -- Purves W. K., et al, "Life: The Science of

---------------------------

End of C.S.M.P. Digest
**********************