From: pottier@clipper.ens.fr (Francois Pottier) Subject: csmp-digest-v3-054 Date: Wed, 7 Sep 1994 18:19:19 +0200 (MET DST) C.S.M.P. Digest Wed, 07 Sep 94 Volume 3 : Issue 54 Today's Topics: How to use AESuspendTheCurrentEvent? Learning Macintosh Programming Should function order be conserved? (CW vs. GestaltValue) Sound Manager headaches (again) Stopping a 'snd ' resource dead [Q] What does "_Xxxxx ,Sys,Immed" mean? jGNE from an INIT 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. To search back issues with WAIS, use comp.sys.mac.programmer.src. With Mosaic, use http://www.wais.com/wais-dbs/comp.sys.mac.programmer.html. ------------------------------------------------------- >From phaedrus@chinook.halcyon.com (Mark Phaedrus) Subject: How to use AESuspendTheCurrentEvent? Date: 19 Aug 1994 21:44:57 GMT Organization: NWNEXUS, Inc. - Making Internet Easy Could someone please post some brief sample code showing the correct use of AESuspendTheCurrentEvent and AEResumeTheCurrentEvent? Basically, this is what I'm doing right now: AppleEvent *pendingEvent,*pendingReply; OSErr err; pascal OSErr MyEventHandler(AppleEvent *messagein, AppleEvent *reply, long refcon) { /* make sure we're allowed to put up a window */ if (noErr != (err = AEInteractWithUser(kMyTimeoutTicks,nil,nil))) return err; /* assorted code to display the window deleted */ /* keep track of these events for later */ pendingEvent = messagein; pendingReply = reply; /* we'll wait for the user to do something that triggers a response */ if (noErr != (err = AESuspendTheCurrentEvent(messagein))) return err; return noErr; } pascal OSErr MyEventCompletion(AppleEvent *messagein,AppleEvent *reply, long refcon) { AEPutParamPtr(reply,keyAEResult,typeChar,(Ptr)myStuff,strlen(myStuff)); return noErr; } void CodeCalledWhenReadyToFinish(void) { err = AEResumeTheCurrentEvent(pendingEvent,pendingReply,MyEventCompletion, 0); } When the event handler is called, AESuspendTheCurrentEvent returns noErr, and the event is apparently suspended correctly; at least, the caller never gets a reply, even though the event handler returns. The problem is that the event and reply records seem to be getting disposed immediately; pendingEvent and pendingReply are pointing at garbage when CodeCalledWhenReadyToFinish is called, and as a result AEResumeTheCurrentEvent fails miserably. Any help you can give is gratefully appreciated... -- \o\ If you're interested in books/stories with transformation \o\ \o\themes, or in furry/anthropomorphic art, email me, or anonymous-\o\ \o\ftp to ftp.halcyon.com and check the /local/phaedrus directory. \o\ \o\ Web users: Now try http://www.halcyon.com/phaedrus/Menu.html \o\ +++++++++++++++++++++++++++ >From hanrek@cts.com (Mark Hanrek) Date: 22 Aug 1994 04:54:34 GMT Organization: The Information Workshop In article <33394p$cs5@news.halcyon.com>, phaedrus@chinook.halcyon.com (Mark Phaedrus) wrote: > Could someone please post some brief sample code showing the correct use > of AESuspendTheCurrentEvent and AEResumeTheCurrentEvent? > Basically, this is what I'm doing right now: There is some example source code called "SuspendAppleEvent" that will hopefully help. You will find under "Inter-Application Communication" within the Snippets collection. It is on the Developer CD and hopefully available at ftp://ftp.apple.com/ as well. Mark Hanrek +++++++++++++++++++++++++++ >From bb@lightside.com (Bob Bradley) Date: Sun, 21 Aug 1994 04:13:05 -0800 Organization: SS Software Inc. I don't have a question of how to use AESuspendTheCurrentEvent but, whether I should use it. When I receive an OpenDoc event, I put up a dialog and the files attached to the OpenDoc event do not get processed until the user hits ok in the dialog (or cancel which would not process the documents at all). Should I suspend the OpenDoc event and resume when (or if) the user chooses ok in the dialog or should I just let the AppleEvent return and process the documents when the user wants? IM Interapp Comm says the AppleEvent Manager expects the event to have been processed when the handler returns which it wouldn't have been in my case and I'm not sure if I should call AESuspendTheCurrentEvent (and resume it later). It works fine though. +++++++++++++++++++++++++++ >From paul@architecture.mcgill.ca (Paul Lalonde) Date: Tue, 23 Aug 1994 02:28:45 GMT Organization: McGill University School of Architecture In article <33394p$cs5@news.halcyon.com>, phaedrus@chinook.halcyon.com (Mark Phaedrus) wrote: > Could someone please post some brief sample code showing the correct use > of AESuspendTheCurrentEvent and AEResumeTheCurrentEvent? > Basically, this is what I'm doing right now: > [ stuff deleted ] You have to save the actual event and reply records, not just pointers to them. So a revised version of the snippet you posted would look like this: AppleEvent pendingEvent,pendingReply; pascal OSErr MyEventHandler(AppleEvent *messagein, AppleEvent *reply, long refcon) { /* stuff deleted */ pendingEvent = *messagein; pendingReply = *reply; /* more stuff deleted */ return noErr; } void CodeCalledWhenReadyToFinish(void) { err = AEResumeTheCurrentEvent(&pendingEvent,&pendingReply, MyEventCompletion,0); } Hope this helps, Paul Lalonde paul@architecture.mcgill.ca --------------------------- >From dchakrav@netserv.unmc.edu (Dhruba Chakravarti) Subject: Learning Macintosh Programming Date: 19 Aug 1994 11:23:44 -0500 Organization: University of Nebraska Medical Center, Omaha, NE, USA I do not know anything about programming, which should be obvious from the header. I have a Macintosh computer and would like to learn programming for Macintosh. Please advise me. Regards, Dhruba. +++++++++++++++++++++++++++ >From mathews@ns9000.furman.edu (Owen Mathews) Date: 19 Aug 1994 18:50:13 GMT Organization: furman university computing center Dhruba Chakravarti (dchakrav@netserv.unmc.edu) wrote: : I do not know anything about programming, which should be obvious : from the header. I have a Macintosh computer and would like to : learn programming for Macintosh. Please advise me. : Regards, : Dhruba. If you're programming in Pascal, pick up the -Macintosh Pascal Programming Primer by Dave Mark and ??? Reed (something like that) **good introductory Mac book; teaches all about the Toolbox and Mac programming fundamentals; comes with several well-thought-out example programs to get you started If you're using C, then try -Macintosh C Programming Primer same fellow(s?) **I haven't see this one, but I assume it's the same idea. If you're really ambitious, take a look at -Learn C++ on the Macintosh guess who? (I'm a real admirer of this guy) **Also probably the same basic idea. No matter which language you use, you'll eventually want to invest in -Inside Macintosh Apple Computer **This set of reference books is the ultimate authority on all aspects of Mac programming. It comes in two flavors: Old Inside Mac (IM) and New IM (NIM). Old IM comes in 6 volumes that are organized chronologically. This unfortunately means that certain related material can be spread out over multiple volumes, forcing you to spend extra time to find the right volume. NIM groups material logically by subject. Each volume covers some clearly-defined area of the Mac API. Of course, NIM's disadvantage is that when Apple releases new stuff, you can't go and pick up the next chronological volume -- instead, you'll probably have to go and buy half a dozen updates to the subject volumes. But I'm rambling. Hey, don't let all that talk daunt you. With any of the first three books, you can get started programming quickly and produce your first (albeit ultra- simple) Mac application within an hour or two. Hope this helps! -- Owen Mathews mathews@furman.edu <><><><><><><><><><><><><><><><><><><><> Furman University, Computer Science Dept +++++++++++++++++++++++++++ >From dchakrav@netserv.unmc.edu (Dhruba Chakravarti) Date: 19 Aug 1994 17:17:42 -0500 Organization: University of Nebraska Medical Center, Omaha, NE, USA Owen Mathews (mathews@ns9000.furman.edu) wrote: : Dhruba Chakravarti (dchakrav@netserv.unmc.edu) wrote: : : I do not know anything about programming, which should be obvious : : from the header. I have a Macintosh computer and would like to : : learn programming for Macintosh. Please advise me. : : Regards, : : Dhruba. : If you're programming in Pascal, pick up the : -Macintosh Pascal Programming Primer : by Dave Mark and ??? Reed (something like that) : **good introductory Mac book; teaches all about the Toolbox : and Mac programming fundamentals; comes with several well-thought-out : example programs to get you started : If you're using C, then try : -Macintosh C Programming Primer : same fellow(s?) : **I haven't see this one, but I assume it's the same idea. : If you're really ambitious, take a look at : -Learn C++ on the Macintosh : guess who? (I'm a real admirer of this guy) : **Also probably the same basic idea. : No matter which language you use, you'll eventually want to invest in : -Inside Macintosh : Apple Computer : **This set of reference books is the ultimate authority on all : aspects of Mac programming. It comes in two flavors: Old Inside Mac : (IM) and New IM (NIM). Old IM comes in 6 volumes that are organized : chronologically. This unfortunately means that certain related material : can be spread out over multiple volumes, forcing you to spend extra : time to find the right volume. NIM groups material logically by : subject. Each volume covers some clearly-defined area of the Mac API. : Of course, NIM's disadvantage is that when Apple releases new stuff, : you can't go and pick up the next chronological volume -- instead, : you'll probably have to go and buy half a dozen updates to the subject : volumes. : But I'm rambling. : Hey, don't let all that talk daunt you. With any of the first three books, : you can get started programming quickly and produce your first (albeit ultra- : simple) Mac application within an hour or two. : Hope this helps! : -- : Owen Mathews mathews@furman.edu : <><><><><><><><><><><><><><><><><><><><> : Furman University, Computer Science Dept Thank you very much! I'll get these books and start reading. Regards, Dhruba. +++++++++++++++++++++++++++ >From Chris Kueny <yun@fusion.ph.utexas.edu> Date: 19 Aug 1994 23:37:21 GMT Organization: The University of Texas at Austin, Austin, Texas In article <332mag$2jq@netserv.unmc.edu> Dhruba Chakravarti, dchakrav@netserv.unmc.edu writes: >I do not know anything about programming, which should be obvious >from the header. I have a Macintosh computer and would like to >learn programming for Macintosh. Please advise me. The C Programming Primer by Dave Mark assumes that you know something about the C programming language (and I assume that the Pascal Programming Primer assumes knowledge of Pascal.) Another good book at this level is Dan Parks Sydow's "Macintosh Programming Techniques". But if you have no programming experience at all, then a good one to start with is "Learn C on the Macintosh" by Dave Mark. This has a stripped down version of Symantec's Think C compiler that can be used to run the projects in the book. It will give you an introduction to the C language, but won't tell you everything. It has a good explanation of programming concepts for people who have never done such things before. As mentioned above, Dave also wrote "Learn C++ on the Macintosh". A classic reference for the whole C language is "The C Programming Language" by Kernighan & Ritchie. I don't have much knowledge of Pascal. I think I heard someone say that it is easier for a beginner to learn Pascal than to learn C. Eventually you will have to buy a compiler. C and Pascal compilers are available at good discounts for university and college students and staff (maybe younger students too?) Used ones are also sometimes advertised in this newsgroup. +++++++++++++++++++++++++++ >From millsp@govonca.gov.on.ca (Phil Mills) Date: Sat, 20 Aug 1994 13:05:05 GMT Organization: Government of Ontario In <332mag$2jq@netserv.unmc.edu> dchakrav@netserv.unmc.edu (Dhruba Chakravarti) writes: >I do not know anything about programming, which should be obvious >from the header. I have a Macintosh computer and would like to >learn programming for Macintosh. Please advise me. 1) Decide why you want to program and what kind of program you would like to create. The best choice in tools and information sources will change depending on that decision. (Without an answer to #1, there *is* no item #2. :-) ) +++++++++++++++++++++++++++ >From kenlong@netcom.com (Ken Long) Date: Sat, 20 Aug 1994 16:52:14 GMT Organization: NETCOM On-line Communication Services (408 261-4700 guest) Learn C on the Macintosh is NOT a good book for someone who has no programming or Mac programming experience. It contains LITTLE about what new Mac programmers are interested in. Think THINK C!, by Dan Parks Sydow is THE book to start with. It teaches new Mac programmers about aspects of Mac programming they WANT to know about. -Ken- +++++++++++++++++++++++++++ >From silicon@Simplex.NL (Silicon Island Software) Date: 21 Aug 1994 08:28:13 GMT Organization: Simplex Networking Amsterdam, The Netherlands. Dhruba Chakravarti (dchakrav@netserv.unmc.edu) wrote: : I do not know anything about programming, which should be obvious : from the header. I have a Macintosh computer and would like to : learn programming for Macintosh. Please advise me. Why Visual EzC is good for learning Macintosh programming. When you program in EzC, the Layouts of the applications with their interface elements are all created graphically in a MacDraw like Layout Editor. The process has little if anything to do with programming. The Menus of the application are also created in a graphic editor by simply adding menus and menu items in a list. The EzC program code is a C-like code, with all the basic syntax and functionality the same as in normal C. There is admittedly nothing new about creating layouts and menus graphically. - The available 4GL languages all do that, each in their own way. What is new though is, that EzC has a very powerful and well known programming language, and the EzC code can be pre-compiled into C, simply by selecting Precompile from the menu. The resulting C code can then be compiled with for instance MPW and the project can be made into a standalone application. Programs written in EzC are just as fast and small in size as programs written from scratch in C or Pascal. To give the curious an example of how a simple program looks like in EzC we can take as an example a window with a text field and listbox and an accept button. When the user keys in a name in the text field and clicks the accept button the name appears in the listbox. Solving the problem: 1) The interface elements are drawn in the layout editor. 2) A menu for the application is created in the menu editor. 3) The code to do the processing is as follows: inherit "Std:System:Window"; string *arr; void _OpenView() { arr=({}); LoadMenubar("Main"); } void _Click(string name) { if(name=="ok") { arr+=({Get("field","text")}); Set("listbox","array",arr); Set("field","text",""); } } The layout with the program code constitutes an object in EzC. The explanation of the code is simple. In the first line we inherit the standard behaviour of a window. EzC is event driven, and opening a window generates an event. To every EzC event corresponds a function. _OpenView() is one such function. If defined in an object, it is invoked when the event occurs. _Click() is another event handler function. It is invoked on clicking the mouse. The interface elements have names. The text field has the been given the name 'field' and the listbox element is called 'listbox' and the accept button's name is 'ok'. The Set() function sets a property of an interface element with given parameters, and the Get() function fetches the value of a property of an interface element. The code is only a few lines, and this is a functioning application, and can be made standalone. It is our opinion that EzC is a suitable language for learning Macintosh programming. - Try for instance for comparison to teach the beginner to create the above example application with Think C using the TCL library. The programmer does not need to study any textbook on Macintosh programming to be able create simple applications like the above application in EzC, nor does he need to have read a word of the voluminous Inside Macintosh. He does however need to know something about C programming in general and to understand the events of EzC and to have at hand a reference over the main kernel and Library functions in EzC. As the programmer becomes more proficient Visual EzC will grow with him, since it is so strongly related to C. The programmer also gradually becomes interested in acquiring knowledge of the Macintosh Toolbox as he needs to solve more advanced problems. The main advantage of learning Macintosh programming by using EzC is that the beginner can actually create useful standalone applications that he can be proud of within hours. Another important advantage is, that the beginner will quickly learn to appreciate Object Oriented programming as object orientation is so closely woven into the EzC language. A third advantage is, that the beginner will learn the basics of a generally known and accepted language, C. Last but not least. One should not forget that EzC also has a built in relational database engine, so when the beginner wants to create a database program, the tools are right there. A demo of Visual EzC can be found on mac.archive.umich.edu in the /mac/misc/demo/visualezc1.0b4-demo.hqx and on any of the info-mac mirror sites. Silicon Island Software Email: silicon@simplex.nl +++++++++++++++++++++++++++ >From D.Birnie@Regy.Canterbury.ac.nz (Denis A. Birnie) Date: Mon, 22 Aug 1994 11:17:06 +1200 Organization: ISS > I do not know anything about programming, which should be obvious > from the header. I have a Macintosh computer and would like to > learn programming for Macintosh. Please advise me. > Dhruba, The current way to go language-wise is C (or C++). The book which I'd recommend (and which uses C) is Macintosh_Programming_Secrets (2nd edition) by Scott Knaster & Keith Rollin. Shows you all sorts of interesting and neat things as well as giving you a skeleton application to work from. A VERY good book. I'd also suggest Think Reference if you get serious and have a screen larger than the original 9"ers. Denis Birnie -- COBOhoLic! +++++++++++++++++++++++++++ >From hanrek@cts.com (Mark Hanrek) Date: 22 Aug 1994 05:04:10 GMT Organization: The Information Workshop In article <332mag$2jq@netserv.unmc.edu>, dchakrav@netserv.unmc.edu (Dhruba Chakravarti) wrote: > I do not know anything about programming, which should be obvious > from the header. I have a Macintosh computer and would like to > learn programming for Macintosh. Please advise me. > Dhruba, Besides all the excellent suggestions and advice thus far, consider doing the following... Decide on a language, and get the compiler for it. Then, download all the example source code you can find in that language and compile it. If something is too hard to get going, move on to another. When you get something to compile successfully, and it runs, then maybe you can think of an easy way to tweak it or make it do something different. - -- This method is an excellent method, because you will quickly become familiar with your development environment, and the language, and the Macintosh Toolbox (the three things you'll be learning at the same time). But what is great about this approach is that you will be getting a sense of gratification as you go along, and you will always be modifying rather than creating -- these are two very important things in the beginning, and two of the major reasons many people give up. There's no need to overwhelm yourself. Always have a language reference handy. So compile freely downloadable program right and left, have fun, make some changes that make goofy things happen, and next thing you know, you will have subliminally learned a ton of essential things without even noticing it. Good luck to you. Mark Hanrek The Information Workshop +++++++++++++++++++++++++++ >From garyg@oak.circa.ufl.edu Date: 22 Aug 1994 12:46:50 GMT Organization: Center for Instructional and Research Computing Activities Dhruba, In addition to the other advice offered, I'd suggest that you also get: Macintosh Programming Techniques: A Foundation for All Macintosh Programmers by Dan Parks Sydow (1994; M&T Books). It includes a diskette with interactive tutorials, sample code with small built applications (in case you don't have a C compiler yet), and lots of highly useful info. Another great job by DPSydow!! hth, Gary +++++++++++++++++++++++++++ >From dchakrav@netserv.unmc.edu (Dhruba Chakravarti) Date: 23 Aug 1994 11:27:23 -0500 Organization: University of Nebraska Medical Center, Omaha, NE, USA garyg@oak.circa.ufl.edu wrote: : Dhruba, : In addition to the other advice offered, I'd suggest that you also get: : Macintosh Programming Techniques: A Foundation for All Macintosh : Programmers by Dan Parks Sydow (1994; M&T Books). It includes a : diskette with interactive tutorials, sample code with small built : applications (in case you don't have a C compiler yet), and lots : of highly useful info. Another great job by DPSydow!! : hth, Gary Thank you all very much! Regards, Dhruba. --------------------------- >From will@cs.su.oz.au (William Uther) Subject: Should function order be conserved? (CW vs. GestaltValue) Date: 20 Aug 1994 16:19:54 +1000 Organization: Basser Dept of Computer Sciece, Uni of Sydney, Australia Hi, I am using CodeWarrior to write a small application. I want this application to publish a handle. I had an earlier version that did this using the GestaltValue library. This library does not work with CW 3.5. I traced it, and found that the reason it didn't work was because it relied upon the ordering of functions to find the size of a block of memory to allocate. (it subtracted the address of the start of one function from the address of the start of another function to get the size of the first function and hence the amount of memory it had to allocate to copy it to the system heap) CW 3.5 does not respect this ordering when it converts Apple's '.o' file into a library. Is this a problem for MetroWerks (CW SHOULD respect the function ordering), or is this a problem for Apple (DTS shouldn't write libraries that rely on function ordering)?? At the moment it just seems to be a problem for me.... Metrowerks tech support were very good at verifying the problem, unfortunately, since they shifted the blame to Apple they haven't done much. (BTW the library works in THINK C 5.0, 6.0 and 7.0 - I assume it works in MPW as it's an MPW object file) Should I be talking to Apple DTS or Metrowerks tech support? \x/ill :-} will@cs.su.oz.au +++++++++++++++++++++++++++ >From cokenias@netcom.com (Damon Cokenias) Date: Sun, 21 Aug 1994 21:16:36 GMT Organization: NETCOM On-line Communication Services (408 261-4700 guest) - --- I traced it, and found that the reason it didn't work was because it relied upon the ordering of functions to find the size of a block of memory to allocate. (it subtracted the address of the start of one function from the address of the start of another function to get the size of the first function and hence the amount of memory it had to allocate to copy it to the system heap) CW 3.5 does not respect this ordering when it converts Apple's '.o' file into a library. - --- Every time I'm stepping through somebody's INIT trying to find out if it is responsible for crashing my machine, I invariably find this sort of programming. It is sick and immoral. It is absolutely none of the programmer's business where or how their code gets compiled. There are two solutions to the Patch/Gestalt Selector/Shutdown Handler install problem. (Actually, there are more than two, but these are my favorites). ** A ** 1) Call Get1IndResource (INIT, 1) to find yourself (or other appropriate GetResource call) 2) Detach yourself, install all patches, etc. ** B ** 1) Load a code resource containing your patch routines, detach and lock 2) Install patches. I typically use A for quickie jobs that I just want to crank out in a moment or two. It has the disadvantage of leaving the loader code behind. Some of the advantages are: Only one binary to build (nice for THINK and CW users), the patches have access to the same A4 globals the loader code did. This can be usefull. I choose B for lean and mean INITs that I release to the public. In order to initialize the globals for the code resource with values from the loader, I sometimes do the following: 1) Inside the code resource I have an A4 global called 'intialized'. It's initial value is zero. 2) BEFORE installing the patches, I call my code resource, passing a pointer to a structure containing initial values. This can only be done for patches that accept parameters, obviously. 3) If 'initialized' is still false, the code resource knows the passed parameter contains special data and uses that data. 'initialized' is then set to true. 4) From this point on, all calls to the code resource are treated as one would expect them to. Often, this is the most convenient way to tell a patch where to jump to after completion. It is 100% portable on all platforms that allow a standalone code resource to have globals. DO NOT: * assume code gets put anywhere. If you use Codewarrior, the frequent updates and changes should cure you of that fast enough. (What if CW noticed that a particular routine could be called with a short branch instead of a long one? It might re-arrange certain funtions to make them closer to their callers.) -Damon Cokenias . -- Damon Cokenias cokenias@netcom.COM +++++++++++++++++++++++++++ >From h+@nada.kth.se (Jon W{tte) Date: Mon, 22 Aug 1994 14:54:28 +0200 Organization: Royal Institute of Something or other In article <cokeniasCuwLro.J8@netcom.com>, cokenias@netcom.com (Damon Cokenias) wrote: >It is sick and immoral. It is absolutely none of the programmer's business >where or how their code gets compiled. Yes; it definately is. For a well-defined runtime architecture, like the Mac 68k one, you're allowed and sometimes forced to do this. >There are two solutions to the Patch/Gestalt Selector/Shutdown Handler install >problem. (Actually, there are more than two, but these are my favorites). >1) Call Get1IndResource (INIT, 1) to find yourself (or other appropriate > GetResource call) >2) Detach yourself, install all patches, etc. >1) Load a code resource containing your patch routines, detach and lock >2) Install patches. Strangely enough, I use none of these. Instead, I get the address of the first instruction (you can use A0 as well, but it's slightly less safe - someone COULD have jumped to you through another register) Then I call RecoverHandle and DetachResource, and use PC-relative addressing to install my patches (and store data) This requires your INIT to be marked pre-loaded, system heap and locked. >1) Inside the code resource I have an A4 global called 'intialized'. It's A4 globals are much more horrible than comparing addresses of functions. A4 globals require you to always enter through main(), unless you do extra inline assembly, which kind of rules out the case where the loader and the patch are in the same code resource. PS: you can get by the loader-left problem by putting the loader code last, branch to it from the top and then call SetHandleSize. -- Jon Wätte (h+@nada.kth.se), Hagagatan 1, 113 48 Stockholm, Sweden "Don't Deal with a Dragon." +++++++++++++++++++++++++++ >From Jaeger@fquest.com (Brian Stern) Date: 22 Aug 1994 17:00:11 GMT Organization: The University of Texas at Austin, Austin, Texas In article <9668AA7E6CA4.FDF19@klkmac008.nada.kth.se>, h+@nada.kth.se (Jon W{tte) wrote: > A4 globals are much more horrible than comparing addresses of > functions. A4 globals require you to always enter through > main(), unless you do extra inline assembly, which kind of > rules out the case where the loader and the patch are in the > same code resource. > This isn't true. With A4 globals you must enter through main *once*. You save the address of A4 in main. After that you can enter through any entry point you like. Each additional entry point first retrieves A4 and can then access its globals. It restores A4 before exiting. There is no reason you can't have the loader and the patch in the same code resource. > PS: you can get by the loader-left problem by putting the > loader code last, branch to it from the top and then call > SetHandleSize. I agree that this is a neat trick. Unfortunately you can't use it when using A4 global addressing. The globals end up at the end of the code resource. Oh well. > -- > Jon Wätte (h+@nada.kth.se), Hagagatan 1, 113 48 Stockholm, Sweden > "Don't Deal with a Dragon." -- Brian Stern :-{)} Jaeger@fquest.com +++++++++++++++++++++++++++ >From h+@nada.kth.se (Jon W{tte) Date: Tue, 23 Aug 1994 08:54:14 +0200 Organization: Royal Institute of Something or other In article <Jaeger-2208941202170001@slip-2-18.ots.utexas.edu>, Jaeger@fquest.com (Brian Stern) wrote: >This isn't true. With A4 globals you must enter through main *once*. You >save the address of A4 in main. After that you can enter through any >entry point you like. Each additional entry point first retrieves A4 and >can then access its globals. It restores A4 before exiting. There is no >reason you can't have the loader and the patch in the same code resource. How do you retrieve the stored value of A4 without extra inline assembly? Or does Symantec provide the extra assembler to do it for you from anywhere now ?(it was a long time since I last used it...) Cheers, / h+ -- Jon Wätte (h+@nada.kth.se), Hagagatan 1, 113 48 Stockholm, Sweden "Psychotherapist" - "Phycho-The-Rapist" Pure coincidence? You decide! +++++++++++++++++++++++++++ >From Jaeger@fquest.com (Brian Stern) Date: 23 Aug 1994 16:19:46 GMT Organization: The University of Texas at Austin, Austin, Texas In article <9668AA7F69B6.206994@klkmac012.nada.kth.se>, h+@nada.kth.se (Jon W{tte) wrote: > How do you retrieve the stored value of A4 without extra inline > assembly? Or does Symantec provide the extra assembler to do it > for you from anywhere now ?(it was a long time since I last used > it...) > > Cheers, > / h+ > -- > Jon Wätte (h+@nada.kth.se), Hagagatan 1, 113 48 Stockholm, Sweden > "Psychotherapist" - "Phycho-The-Rapist" > Pure coincidence? You decide! Yes, the inline code is provided in a header file that must be included in each source file that needs to do A4 addressing. The header contains several inline assembler functions for saving and restoring A4. From the standpoint of the programmer there is no assembly required. -- Brian Stern :-{)} Jaeger@fquest.com +++++++++++++++++++++++++++ >From phils@bedford.symantec.com (Phil Shapiro) Date: Tue, 23 Aug 1994 19:31:26 -0400 Organization: Symantec Corp. In article <9668AA7F69B6.206994@klkmac012.nada.kth.se>, h+@nada.kth.se (Jon W{tte) wrote: | In article <Jaeger-2208941202170001@slip-2-18.ots.utexas.edu>, | Jaeger@fquest.com (Brian Stern) wrote: | | >This isn't true. With A4 globals you must enter through main *once*. You | >save the address of A4 in main. After that you can enter through any | >entry point you like. Each additional entry point first retrieves A4 and | >can then access its globals. It restores A4 before exiting. There is no | >reason you can't have the loader and the patch in the same code resource. | | How do you retrieve the stored value of A4 without extra inline | assembly? Or does Symantec provide the extra assembler to do it | for you from anywhere now ?(it was a long time since I last used | it...) Each source file gets its own copy of A4. As long as you're in the same source file that you called RememberA0() in (or RememberA4(), if A4 was valid already), you can call SetUpA4() at any entry point. This is because A4 is stored inside code in a static function; each file that uses the function gets a different copy. This has been true for quite awhile, I'm pretty sure that LSC 4.0 worked this way. If you build with a custom header, you don't even need to enter main() at all. But you better set up A4 before you access any globals or cross-segment routines. -phil --------------------------- >From "Andrew C. Plotkin" <ap1i+@andrew.cmu.edu> Subject: Sound Manager headaches (again) Date: Sun, 21 Aug 1994 14:42:43 -0400 Organization: Information Technology Center, Carnegie Mellon, Pittsburgh, PA When we last left our hero, he had just figured out how to play a sound at a given frequency (by sending soundCmd and freqDurationCmd to a sampledSynth channel.) (I sent a code snippet of doing this to alt.sources.mac, if you care. ./vol-08/sampledsnd.cpt.hqx) Well, there was a reason I wanted to do that. See, I want to get music out, and I thought that the way to do that was record snd resources of each instrument, and then use that technique to play notes. Allocate several channels, you can get multiple notes at once. ...Riiiiight. First problem: they don't start off in synch. I use the algorithm given in the PlayTwoSoundsSynched in the SM3.0 tech note: (warning: pseudocode, not real code, ahead) SndDoImmediate(chan1, syncCmd{3, kID}) SndDoCommand(chan1, restCmd{500}); SndDoCommand(chan1, freqDurationCmd{500, MIDDLE_C}); SndDoImmediate(chan2, syncCmd{3, kID}) SndDoCommand(chan2, restCmd{500}); SndDoCommand(chan2, freqDurationCmd{500, MIDDLE_G}); SndDoImmediate(chan1, syncCmd{1, kID}) I expect this to pause a quarter-second, then play a nice chord. Nope. One channel (the first one, I believe) starts slightly -- but audibly -- first. (They do finish together, though.) This occurs both with and without SM3.0. (My guess: the SM has to resample one or both sounds to a different frequency, and this takes some time. Sigh.) Oh, yeah -- syncCmd behaves with extreme magic. I still don't know exactly what it does. For example, if I change the last line in the algorithm above from chan1 to chan2, only one note plays instead of both. Huh? Second problem: Under SM3.0, if I play a sequence of notes of the same duration, they don't come out with the same duration. Again, the difference is pretty audible -- as much as a 50% error in short notes. Unless I can do something about this, my entire plan is screwed. (Kinda cool, huh? The nifty new SM *doesn't support freqDurationCmd* in a usable form. I was pleased.) I recall someone else posted about this a few weeks ago, and he couldn't get it to work either. (My guess: same reason as before. The SM *does* keep the same (distorted) time between channels; if I start two parallel synched sequences, they stay synched. I guess that's something. It still ruins the music, though.) My question, pretty much, is "Have I missed anything?" I don't think I have. I think I'm going to have to ditch the whole idea of freqDurationCmd, and write my *own* sound-resampling-and-playing module a la SoundTracker. (Or just use SoundTracker or some other MOD player. This is tempting, but I haven't seen a freeware MOD player which includes source code, except for an icky-looking one ported straight from Unix, with malloc() everywhere. Anyone? Anyone?) ObBribery: if you give me a magic answer that fixes my music-playing code, I'll release the code as freeware. And I'll throw in the simple-but-oh-so-elegant music editor I've spent the past month on. (OTOH, if I wind up writing a MOD-style thing, I'll do the same thing. Nevertheless.) --Z "And Aholibamah bare Jeush, and Jaalam, and Korah: these were the borogoves..." +++++++++++++++++++++++++++ >From Rick_Holzgrafe@taligent.com (Rick Holzgrafe) Date: Mon, 22 Aug 1994 18:22:58 GMT Organization: Semicolon Software In article <siJu0X_00gpI9iY0YJ@andrew.cmu.edu>, "Andrew C. Plotkin" <ap1i+@andrew.cmu.edu> wrote: > Second problem: Under SM3.0, if I play a sequence of notes of the same > duration, they don't come out with the same duration. Again, the > difference is pretty audible -- as much as a 50% error in short notes. > Unless I can do something about this, my entire plan is screwed. (Kinda > cool, huh? The nifty new SM *doesn't support freqDurationCmd* in a > usable form. I was pleased.) I recall someone else posted about this a > few weeks ago, and he couldn't get it to work either. That was me, I think. I've heard The Word from The Source: a key programmer at Apple tells me that SM 3.0 has been optimized for playing sampled sounds. The optimizations DE-optimized for playing melodies via freqDurationCmd; something about internal buffers having a size that causes perceptible delays when they need to be refilled. There is no workaround. For my app, I had to give up on freqDurationCmds (under SM 3.0) and convert my tunes to single sampled sound resources, which increased their size by two orders of magnitude and (to keep them from getting even larger) decreased their sound quality. My app now tests the version of the Sound Manager; if it is 3.0, you get one and only one melody; if less than 3.0, you get a rotating selection out of a list of seven melodies. The seven melodies take up about 20K of disk space; the one sampled-sound melody takes several times that. Feh. FWIW, I can show you code that should solve your synchronization problem. Your various melodic parts will start together correctly, and they will stay synched. Doesn't help the uneven tempo, though. -- Rick Holzgrafe Semicolon Software rmh@taligent.com +++++++++++++++++++++++++++ >From h+@nada.kth.se (Jon W{tte) Date: Tue, 23 Aug 1994 09:49:07 +0200 Organization: Royal Institute of Something or other In article <Rick_Holzgrafe-2208941123020001@ricks-cafe.taligent.com>, Rick_Holzgrafe@taligent.com (Rick Holzgrafe) wrote: >version of the Sound Manager; if it is 3.0, you get one and only one >melody; if less than 3.0, you get a rotating selection out of a list of >seven melodies. The seven melodies take up about 20K of disk space; the >one sampled-sound melody takes several times that. Feh. With QuickTime 2.0, however, all you do is store a General MIDI file in your resource fork (3-10k) and there's your melody. Luckily, this isn't rec.music.makers.synth, where "General MIDI" is a cuss word. Cheers, / h+ -- Jon Wätte (h+@nada.kth.se), Hagagatan 1, 113 48 Stockholm, Sweden "Psychotherapist" - "Phycho-The-Rapist" Pure coincidence? You decide! +++++++++++++++++++++++++++ >From Rick_Holzgrafe@taligent.com (Rick Holzgrafe) Date: Tue, 23 Aug 1994 18:24:34 GMT Organization: Semicolon Software In article <Rick_Holzgrafe-2208941123020001@ricks-cafe.taligent.com>, Rick_Holzgrafe@taligent.com (Rick Holzgrafe) wrote: >version of the Sound Manager; if it is 3.0, you get one and only one >melody; if less than 3.0, you get a rotating selection out of a list of >seven melodies. The seven melodies take up about 20K of disk space; the >one sampled-sound melody takes several times that. Feh. In article <9668AA7F7693.236F39@klkmac012.nada.kth.se>, h+@nada.kth.se (Jon W{tte) wrote: > With QuickTime 2.0, however, all you do is store a General MIDI > file in your resource fork (3-10k) and there's your melody. Great. So now instead of checking for the version of the Sound Manager and having two methods of playing music, I must also check for the presence and version of QuickTime, and have THREE methods. Ain't progress wonderful? Your pardon, please, for my surliness. I'm just tired of carefully coding to Apple's published interfaces, and having my sound code break anyway every couple of years. -- Rick Holzgrafe Semicolon Software rmh@taligent.com --------------------------- >From mdc14@ciao.cc.columbia.edu (Michael D Coble) Subject: Stopping a 'snd ' resource dead Date: 23 Aug 1994 03:53:01 GMT Organization: Columbia University I am a programming student trying to write a couple of macintosh applications. I recorded some music into a 'snd ' resource and want to invoke the music when a user chooses the about this application from the apple menu accompanied by a little animation. The 'snd ' resource is a couple of minutes long, so I'd like to be able to allow a user to click the close box and stop it in its tracks. I wonder if one of you experienced programmers out there might point me in the direction to reach this goal. Thanks, Michael Coble, mdc14@columbia.edu +++++++++++++++++++++++++++ >From bb@lightside.com (Bob Bradley) Date: Sun, 21 Aug 1994 15:12:37 -0800 Organization: SS Software Inc. In article <33brqt$mi8@apakabar.cc.columbia.edu>, mdc14@ciao.cc.columbia.edu (Michael D Coble) wrote: > I am a programming student trying to write a couple of macintosh > applications. I recorded some music into a 'snd ' resource and want to > invoke the music when a user chooses the about this application from the > apple menu accompanied by a little animation. The 'snd ' resource is a > couple of minutes long, so I'd like to be able to allow a user to click > the close box and stop it in its tracks. I wonder if one of you > experienced programmers out there might point me in the direction to > reach this goal. You can do this by playing the sound async. Something like this: SndNewChannel(...) // Create a new sound channel soundHandle = GetResource(...) // Get the snd resource SndPlay(...) // Play the sound setting async to true SndDisposeChanne(...) // Dispose of the sound channel and setting // quietNow to true so it stops any sound // that is currently playing. Hope that helps and doesn't contain too many errors :) +++++++++++++++++++++++++++ >From reed@medicine.wustl.edu (Thomas Reed) Date: Tue, 23 Aug 1994 11:08:58 -0500 Organization: Washington University If you ftp to thomas_mac.wustl.edu, you'll find the source code for a solitaire game I wrote a while back. It contains some very useful libraries I've written -- one of which is a sound library. Not only will it let you do what you want to do, but it'll also let you play the sound from disk to reduce the memory requirements. Hope you like it! Oh, for those of you with NewsWatcher, here's the URL: ftp://thomas_mac.wustl.edu/Public/for_ftp/Solitaire_code.sit -Thomas ===================================================== Thomas Reed Washington University reed@telesphere.wustl.edu Medical School reed@medicine.wustl.edu Saint Louis, MO - --------------------------------------------------- Clothes make the man. Naked people have little or no influence on society. -- Mark Twain ===================================================== Opinions posted are not the opinions of Wash. U. --------------------------- >From chris-b@cs.auckland.ac.nz (Chris Burns) Subject: [Q] What does "_Xxxxx ,Sys,Immed" mean? Date: Wed, 24 Aug 1994 11:36:07 +1200 Organization: AucklandUniversity:ComputerScience:HMU Hi all, What do the ",Sys,Immed" qualifiers mean? I know they specify certain bits in the trap word, but what do the different combinations mean? What I want is a table: Sys Immed Meaning x x ? x ? x ? ? Thanks Heaps In Advance, Chris B - --------------------------------------------------------------------- NewZealand:AucklandUniversity:ComputerScience:HyperMediaUnit:ChrisBurns Internet: chris-b@cs.auckland.ac.nz Phone: +64 9 373-7599 x6194 Fax: +64 9 373-7453 Async, therefore I am. - --------------------------------------------------------------------- +++++++++++++++++++++++++++ >From wysocki@netcom.com (Chris Wysocki) Date: Wed, 24 Aug 1994 15:27:25 GMT Organization: NETCOM On-line Communication Services (408 261-4700 guest) In article <chris-b-2408941136070001@hmu7.cs.aukuni.ac.nz>, Chris Burns <chris-b@cs.auckland.ac.nz> wrote: >What do the ",Sys,Immed" qualifiers mean? >I know they specify certain bits in the trap word, but what do the >different combinations mean? The ",Sys" and ",Immed" qualifiers displayed by MacsBug can be misnomers, since these are only two of the possible meaning of bits 9 and 10 of a trap word. All the possible meaning for these bits can be found in Traps.a: ; Equates for setting trap option bits ; for Device and File Manager routines immed EQU $200 ; execute immediately, bypass I/O queue async EQU $400 ; asynchronous, don't wait for completion ; for Memory Manager routines clear EQU $200 sys EQU $400 ; for string routines marks EQU $200 ; set to ignore/strip diacriticals case EQU $400 ; set for case sensitivity ; for all Toolbox routines autoPop EQU $400 ; set to pop an extra return address ; for Get/Set & NGet/NSet TrapAddress newTool EQU $600 ; Toolbox trap, under new ordering newOS EQU $200 ; OS trap, under new ordering ; for HFS routines newHFS EQU $200 Chris. --------------------------- >From chrism@col.hp.com (Chris Magnuson) Subject: jGNE from an INIT Date: 17 Aug 1994 05:50:18 GMT Organization: HP Colorado Springs Division I am writing a jGNE event filter in MPW that needs to be part of an INIT that does alot of other things ( i.e. the jGNE function is one part of the INIT code resource). I've gotten alot done, but run into problems when events actually call my jGNE routine to do the event "interception/ redirection." short JGNEFilterFunc(short hasEvent, const EventRecord *theEvent) My question is this: when the JGNEFilterFunc routine is called with an EventRecord, etc. am I supposed to be doing some A5 world swapping to get things to be handled properly in the jGNE routine? Without posting alot of code here is basically what the INIT initialization does. Hints are greatly appreciated! main() { oldZone = GetZone(); SetZone(SystemZone()); MakeA5World(&A5Ref); if (A5Ref == nil) { SetZone(oldZone); return(memFullErr); } a_SetA5Ref(A5Ref); oldA5 = SetA5World(A5Ref); (void) ReadPrefs(); jGNEInit(); /* point to assembly glue routine that eventually calls * JGNEFilterFunc() when events go through jGNE 0x29A */ (void) SetA5(oldA5); SetZone(oldZone); } Chris Magnuson chrism@col.hp.com +++++++++++++++++++++++++++ >From gurgle@dnai.com (Pete Gontier) Date: Tue, 16 Aug 1994 23:10:27 -0800 Organization: Integer Poet Software In article <32s8eq$kdr@hp-col.col.hp.com>, chrism@col.hp.com (Chris Magnuson) wrote: > I am writing a jGNE event filter in MPW that needs to be part of an INIT... > am I supposed to be doing some A5 world swapping... ? You will want to read the DSC Technical Note "Stand-alone Code Resources Ad Nauseum" if you have not already. The short answer is yes, you do need to do some twiddling to get access to global variables. Frankly, though, I would recommend doing this project in THINK C, because (a) assembly code is easier to integrate, (b) global variables are easier to access, and (c) my sample jGNE INIT, "jGNE Helper", is posted monthly to 'alt.sources.mac' with a THINK C project and will make all of these problems go away for you. -- Pete Gontier // CTO, Integer Poet Software // gurgle@dnai.com "I am Homer of Borg! Prepare to be... Ooooooo! Donuts!" +++++++++++++++++++++++++++ >From neil_ticktin@xplain.com (Neil Ticktin) Date: Wed, 17 Aug 1994 16:26:42 GMT Organization: MacTech Magazine/Xplain Corporation In article <32s8eq$kdr@hp-col.col.hp.com>, chrism@col.hp.com (Chris Magnuson) wrote: >> I am writing a jGNE event filter in MPW that needs to be part of an INIT that >> does alot of other things ( i.e. the jGNE function is one part of the >> INIT code resource). I've gotten alot done, but run into problems when >> events actually call my jGNE routine to do the event "interception/ >> redirection." >> Chris, Late last year, there was a great article by Symantec Tech Support in about writing a jGNEFilter. It was one of the THINK Top 10 columns. It contained the template code and an example of how to do it. I'm on the road and don't have the issue handy, but you should check it out. Hope it helps, Neil - --------------------------------------------------------------------- Neil Ticktin, MacTech Magazine (formerly MacTutor) PO Box 250055, Los Angeles, CA 90025 * 310-575-4343 * Fax: 310-575-0925 For more info, anonymous ftp to ftp.netcom.com and cd to /pub/xplain custservice@xplain.com * editorial@xplain.com * adsales@xplain.com marketing@xplain.com * accounting@xplain.com * pressreleases@xplain.com progchallenge@xplain.com * publisher@xplain.com * info@xplain.com +++++++++++++++++++++++++++ >From chrism@col.hp.com (Chris Magnuson) Date: 17 Aug 1994 20:14:57 GMT Organization: HP Colorado Springs Division Pete Gontier (gurgle@dnai.com) wrote: : In article <32s8eq$kdr@hp-col.col.hp.com>, : chrism@col.hp.com (Chris Magnuson) wrote: : > I am writing a jGNE event filter in MPW that needs to be part of an INIT... : > am I supposed to be doing some A5 world swapping... ? : You will want to read the DSC Technical Note "Stand-alone Code Resources : Ad Nauseum" if you have not already. The short answer is yes, you do need : to do some twiddling to get access to global variables. : Frankly, though, I would recommend doing this project in THINK C, because : (a) assembly code is easier to integrate, (b) global variables are easier : to access, and (c) my sample jGNE INIT, "jGNE Helper", is posted monthly : to 'alt.sources.mac' with a THINK C project and will make all of these : problems go away for you. No it won't :) I must not be communicating what has to happen here. main() is no longer the entry point for a jGNE filter only. main() is the entry point for an entire INIT, one component of which is a jGNE event processor. So the code resource in question here is an entire INIT, not just a jGNE filter. In other words, the jGNE portion of the code is one routine within the INIT code resource, and I have glue that dispatches to that routine, but it does not behave properly. So maybe the question is, when the jGNE event is dispatched to the code within my INIT, must I set A5 for some reason? Even though I am not interested in globals (just the parameters on the stack)? Chris Magnuson chrism@col.hp.com +++++++++++++++++++++++++++ >From chrism@col.hp.com (Chris Magnuson) Date: 17 Aug 1994 20:16:09 GMT Organization: HP Colorado Springs Division Thanks. Yes, I did check that out. It doesnt answer my question since it is a standalone jGNE (in other words the jGNE event processor is not part of a larger code resource with other things going on). Chris Magnuson chrism@col.hp.com Neil Ticktin (neil_ticktin@xplain.com) wrote: : In article <32s8eq$kdr@hp-col.col.hp.com>, chrism@col.hp.com (Chris : Magnuson) wrote: : >> I am writing a jGNE event filter in MPW that needs to be part of an : INIT that : >> does alot of other things ( i.e. the jGNE function is one part of the : >> INIT code resource). I've gotten alot done, but run into problems when : >> events actually call my jGNE routine to do the event "interception/ : >> redirection." : >> : Chris, : Late last year, there was a great article by Symantec Tech Support in : about writing a jGNEFilter. It was one of the THINK Top 10 columns. It : contained the template code and an example of how to do it. I'm on the : road and don't have the issue handy, but you should check it out. : Hope it helps, : Neil : ----------------------------------------------------------------------- : Neil Ticktin, MacTech Magazine (formerly MacTutor) : PO Box 250055, Los Angeles, CA 90025 * 310-575-4343 * Fax: 310-575-0925 : For more info, anonymous ftp to ftp.netcom.com and cd to /pub/xplain : custservice@xplain.com * editorial@xplain.com * adsales@xplain.com : marketing@xplain.com * accounting@xplain.com * pressreleases@xplain.com : progchallenge@xplain.com * publisher@xplain.com * info@xplain.com +++++++++++++++++++++++++++ >From larson@oahu.cs.ucla.edu (Christopher Larson) Date: Wed, 17 Aug 94 21:28:50 GMT Organization: UCLA, Computer Science Department In article <32tr42$pfg@hp-col.col.hp.com> chrism@col.hp.com (Chris Magnuson) writes: >: In article <32s8eq$kdr@hp-col.col.hp.com>, >: chrism@col.hp.com (Chris Magnuson) wrote: > >: > I am writing a jGNE event filter in MPW that needs to be part of an INIT... >: > am I supposed to be doing some A5 world swapping... ? [...] > So maybe the question is, when the jGNE event is dispatched to the code > within my INIT, must I set A5 for some reason? Even though I am not > interested in globals (just the parameters on the stack)? If you don't use (or cause to be used) any globals, you don't need an A5 world, and, therefore, you don't need to set A5. If the jGNE filter doesn't need access to any globals, why is it part of the INIT (I read the above to mean the jGNE code is within the INIT resource, if not, ignore this)? Wouldn't it be easier to compile the jGNE portion as a seperate resource, and just load, detatch, and install from the INIT resource? --Chris <Wondering if he's missing something obvious> Larson _______________________________________________________________________________ Chris Larson -- Amateur Macintosh Geek, CoBase Research Assistant L.A. Institute of Slowly and Painfully Working Out the Surprisingly Obvious Death to the Trojans! Go Bruins! (Insert disclaimer here) Internet: larson@kingston.cs.ucla.edu +++++++++++++++++++++++++++ >From Jaeger@fquest.com (Brian Stern) Date: 18 Aug 1994 00:53:36 GMT Organization: The University of Texas at Austin, Austin, Texas In article <32tr42$pfg@hp-col.col.hp.com>, chrism@col.hp.com (Chris Magnuson) wrote: > Pete Gontier (gurgle@dnai.com) wrote: > : In article <32s8eq$kdr@hp-col.col.hp.com>, > : chrism@col.hp.com (Chris Magnuson) wrote: > > : > I am writing a jGNE event filter in MPW that needs to be part of an INIT... > : > am I supposed to be doing some A5 world swapping... ? > > No it won't :) I must not be communicating what has to happen here. main() > is no longer the entry point for a jGNE filter only. main() is the entry > point for an entire INIT, one component of which is a jGNE event processor. > So the code resource in question here is an entire INIT, not just a jGNE > filter. > > In other words, the jGNE portion of the code is one routine within the INIT > code resource, and I have glue that dispatches to that routine, but it > does not behave properly. > > So maybe the question is, when the jGNE event is dispatched to the code > within my INIT, must I set A5 for some reason? Even though I am not > interested in globals (just the parameters on the stack)? > > Chris Magnuson > chrism@col.hp.com In order to function properly any jGNEFilter has to insert itself into the chain of all jGNEFilters. When your filter exits it needs to jump to the previous filter. The address of the previous filter must be saved at the time that you install your filter. This saved address must be a global. Therefore your jGNEFilter, and all jGNEFilters for all time, must have access to global variables. There are several mechanisms for accessing global variables in stand alone code resources like your jGNEFilter. As I understand it MPW uses an A5-based mechanism that is described in the tech note. Use that mechanism if you like. Think, and I believe CW, use an A4-based mechanism, which is straightforward to use. Aside from access to the previous filter's address or to other global variables there is no requirement for an A5 world to be set up. There is a sample jGNEFilter written by Dave Falkenberg that can be found at ftp.apple.com. This sample is all in MPW assembler. It does not use the A5 mechanism for storing the address of the previous filter. It simply stores the address within itself. -- Brian Stern :-{)} Jaeger@fquest.com +++++++++++++++++++++++++++ >From gurgle@dnai.com (Pete Gontier) Date: Wed, 17 Aug 1994 21:17:15 -0800 Organization: Integer Poet Software In article <1994Aug17.212850.10867@cs.ucla.edu>, larson@oahu.cs.ucla.edu (Christopher Larson) wrote: > If the jGNE filter doesn't need access to any globals, why is it part of > the INIT ... ? Wouldn't it be easier to compile the jGNE portion as > a seperate resource, and just load, detatch, and install from the INIT > resource? His filter might want to call other functions in the INIT without duplicating them in the jGNE resource. I don't know enough about MPW-generated code resources to be able to say whether it needs A5 to call functions. Probably it uses PC-relative calls, but it's something you would want to look up. -- Pete Gontier // CTO, Integer Poet Software // gurgle@dnai.com "Even during a particularly harsh (Colorado) winter... many of the 300 families in the VCTV (movies-on-demand) trial continued to go to video stores." -- eis@murrow.tufts.edu, in Wired 2.09 p62 +++++++++++++++++++++++++++ >From gurgle@dnai.com (Pete Gontier) Date: Wed, 17 Aug 1994 21:25:33 -0800 Organization: Integer Poet Software In article <32tr42$pfg@hp-col.col.hp.com>, chrism@col.hp.com (Chris Magnuson) wrote: > I must not be communicating what has to happen here. main() > is no longer the entry point for a jGNE filter only. main() is the entry > point for an entire INIT, one component of which is a jGNE event processor. > So the code resource in question here is an entire INIT, not just a jGNE > filter. In other words, the jGNE portion of the code is one routine within > the INIT code resource, and I have glue that dispatches to that routine, > but it does not behave properly. As I said in email, I know. :-) Probably I am the one who did insufficient communicating. With a THINK (or CodeWarrior) compiler, what you're asking about is trivial, because the compiler actually knows that it's compiling something other than an application, unlike MPW. MPW only knows at link time; consequently, MPW code resource have to pretend in many ways that they are apps. Hence that entire tech note about code resources and A5. Blech. > So maybe the question is, when the jGNE event is dispatched to the code > within my INIT, must I set A5 for some reason? Even though I am not > interested in globals (just the parameters on the stack)? Again as I said in email, if your code does not use statically allocated variables (locals declared 'static' or globals), you may still need to set up A5 if your filter calls other routines, depending on how MPW finds those routines. Probably it finds them PC-relative, but it may find them A5-relative. Perhaps some MPW guru can shed light on this. -- Pete Gontier // CTO, Integer Poet Software // gurgle@dnai.com "Even during a particularly harsh (Colorado) winter... many of the 300 families in the VCTV (movies-on-demand) trial continued to go to video stores." -- eis@murrow.tufts.edu, in Wired 2.09 p62 +++++++++++++++++++++++++++ >From Jaeger@fquest.com (Brian Stern) Date: 19 Aug 1994 02:36:45 GMT Organization: The University of Texas at Austin, Austin, Texas In article <jedavis.777254033@Xenon.Stanford.EDU>, jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: > On a side note.. I dont suppose you have any idea if I can > allocate some memory in a jGNEfilter? I dont mind stealing > from the host app. I want to muck with the clipboard, so need > a couple handles that dont mind being resized by the Scrap calls. > > -James Since jGNEFilters are called from WNE/GNE and both of those traps are documented as moving or purging memory, Yes you can allocate memory from inside a jGNEFilter. If you want the memory to last longer than one pass through your filter make sure to allocate the memory in the system heap, otherwise it will be lost if the app quits. If the memory will be allocated and disposed within one pass of the filter it can go in the application's heap. -- Brian Stern :-{)} Jaeger@fquest.com +++++++++++++++++++++++++++ >From jedavis@Xenon.Stanford.EDU (James Edward Davis) Date: 18 Aug 94 18:51:13 GMT Organization: Stanford University: Computer Science Department larson@oahu.cs.ucla.edu (Christopher Larson) writes: >In article <32tr42$pfg@hp-col.col.hp.com> chrism@col.hp.com (Chris Magnuson) writes: >> So maybe the question is, when the jGNE event is dispatched to the code >> within my INIT, must I set A5 for some reason? Even though I am not >> interested in globals (just the parameters on the stack)? >If you don't use (or cause to be used) any globals, you don't need an A5 >world, and, therefore, you don't need to set A5. Yes, but remember that some Quickdraw globals are offset from A5 as well. You might not need *your* A5, but you will need a valid one. I bring this up, because inside your jGNEfilter you wont neccessarily get one. You can set the A5 of the currently executing application from the low memory global CurrentA5, but remember to put back whatever stuffs was in A5 to start with. James Davis : jedavis@cs.stanford.edu +++++++++++++++++++++++++++ >From larson@oahu.cs.ucla.edu (Christopher Larson) Date: Thu, 18 Aug 94 21:05:17 GMT Organization: UCLA, Computer Science Department James said: >I said: >>Chris Magnuson said: >>> So maybe the question is, when the jGNE event is dispatched to the code >>> within my INIT, must I set A5 for some reason? Even though I am not >>> interested in globals (just the parameters on the stack)? > >>If you don't use (or cause to be used) any globals, you don't need an A5 >>world, and, therefore, you don't need to set A5. > >Yes, but remember that some Quickdraw globals are offset from A5 as >well. Making any calls which require Quickdraw globals count as causing globals to be used. (Perhaps I should have made that point clear :-). > You might not need *your* A5, but you will need a valid one. Why? If I don't use Quickdraw or my own globals, I shouldn't (see caveat below) need an A5 world and shouldn't care what the value of A5 is. I've written numerous code resources and trap patches (though never a jGNE filter). I've never needed an A5 world of my own (meaning belonging to the code resource), and only needed to borrow the current application's A5 when using Quickdraw globals. Unless jGNE filters are inherently different, I don't see why one must run with a valid A5. (I didn't intend this to sound at all like a flame. I'm just trying to get to the bottom of this too.) caveat: As Peter Lewis pointed out, calls to other routines within code resources may use a jump table, which may use A5. Consult your development system regarding this. (I'm paraphrasing, excuse me if I've misinterpreted you, Peter.) >I bring this up, because inside your jGNEfilter you wont neccessarily get >one. You sure won't. >You can set the A5 of the currently executing application from the low >memory global CurrentA5, but remember to put back whatever stuffs was in >A5 to start with. An invaluable piece of information. Saved my hiney many times. --Chris _______________________________________________________________________________ Chris Larson -- Amateur Macintosh Geek, CoBase Research Assistant L.A. Institute of Slowly and Painfully Working Out the Surprisingly Obvious Death to the Trojans! Go Bruins! (Insert disclaimer here) Internet: larson@kingston.cs.ucla.edu +++++++++++++++++++++++++++ >From jedavis@Xenon.Stanford.EDU (James Edward Davis) Date: 18 Aug 94 23:53:53 GMT Organization: Stanford University: Computer Science Department larson@oahu.cs.ucla.edu (Christopher Larson) writes: >> You might not need *your* A5, but you will need a valid one. >Why? If I don't use Quickdraw or my own globals, I shouldn't (see caveat >below) need an A5 world and shouldn't care what the value of A5 is. I've >written numerous code resources and trap patches (though never a jGNE >filter). I've never needed an A5 world of my own (meaning belonging to >the code resource), and only needed to borrow the current application's >A5 when using Quickdraw globals. Unless jGNE filters are inherently >different, I don't see why one must run with a valid A5. (I didn't >intend this to sound at all like a flame. I'm just trying to get to >the bottom of this too.) You're correct. I phrased something badly. I had meant to suggest that stealing the current applications A5 would do for quickdraw globals, if you didnt have one of your own handy. On a side note.. I dont suppose you have any idea if I can allocate some memory in a jGNEfilter? I dont mind stealing from the host app. I want to muck with the clipboard, so need a couple handles that dont mind being resized by the Scrap calls. -James +++++++++++++++++++++++++++ >From jedavis@Xenon.Stanford.EDU (James Edward Davis) Date: 19 Aug 94 09:42:08 GMT Organization: Stanford University: Computer Science Department Jaeger@fquest.com (Brian Stern) writes: >In article <32tr42$pfg@hp-col.col.hp.com>, chrism@col.hp.com (Chris >Magnuson) wrote: >> Pete Gontier (gurgle@dnai.com) wrote: >> : In article <32s8eq$kdr@hp-col.col.hp.com>, >> : chrism@col.hp.com (Chris Magnuson) wrote: >> >> So maybe the question is, when the jGNE event is dispatched to the code >> within my INIT, must I set A5 for some reason? Even though I am not >> interested in globals (just the parameters on the stack)? >> >> chrism@col.hp.com >In order to function properly any jGNEFilter has to insert itself into the >chain of all jGNEFilters. When your filter exits it needs to jump to the >previous filter. The address of the previous filter must be saved at the >time that you install your filter. This saved address must be a global. Global? Well not really. Its not really good style, but Ive seen it done by stuffing the address into your code at install time. The same way many people store their app A5, with code that self modifies to contain the location. For that matter, where should I store my apps A5, so I can reset it without using self modifying code? -James +++++++++++++++++++++++++++ >From larson@oahu.cs.ucla.edu (Christopher Larson) Date: Fri, 19 Aug 94 16:15:35 GMT Organization: UCLA, Computer Science Department In article <jedavis.777289328@Xenon.Stanford.EDU> jedavis@Xenon.Stanford.EDU (James Edward Davis) writes: >Jaeger@fquest.com (Brian Stern) writes: >>In order to function properly any jGNEFilter has to insert itself into the >>chain of all jGNEFilters. When your filter exits it needs to jump to the >>previous filter. The address of the previous filter must be saved at the >>time that you install your filter. This saved address must be a global. > >Global? Well not really. Its not really good style, but Ive seen it done >by stuffing the address into your code at install time. The same way >many people store their app A5, with code that self modifies to >contain the location. I'm not sure this strategy counts as self-modification if the installed code is contained in its own resource. When the app (INIT, whatever) loads the resource, before detaching and installing, it's just data. If it's modified at this point, I don't think you will have any problems (still need to flush caches though), because the resource has never been loaded as code. If the code to be installed is in your 'CODE' resources, all bets are off. (Comments on this welcome....) >For that matter, where should I store my apps A5, so I can reset >it without using self modifying code? Well, you could use the above strategy, or if that is still to self-modify- ish, you could install a Gestalt selector. Note that trickery to recover your A5 is only necessary if your code might run when your app is switched out or at interrupt time. If this is not the case, you can recover it from CurrentA5, as you suggested earlier. Hmmm. This raises an interesting question. If an app installs a jGNEFilter, is that filter visible to all other applications, or is it more like trap patches (which, if done from an app, are only visible to the installing app)? Anyone? Since this thread is about jGNEFilters, I assume your question was specific to recovering A5 from a jGNEFilter/trap patch kind of situation. If not, mail me with the specifics. --Chris _______________________________________________________________________________ Chris Larson -- Amateur Macintosh Geek, CoBase Research Assistant L.A. Institute of Slowly and Painfully Working Out the Surprisingly Obvious Death to the Trojans! Go Bruins! (Insert disclaimer here) Internet: larson@kingston.cs.ucla.edu +++++++++++++++++++++++++++ >From larson@oahu.cs.ucla.edu (Christopher Larson) Date: Fri, 19 Aug 94 16:22:01 GMT Organization: UCLA, Computer Science Department In article <Jaeger-1808942138440001@slip-15-7.ots.utexas.edu> Jaeger@fquest.com (Brian Stern) writes: >In article <jedavis.777254033@Xenon.Stanford.EDU>, >jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: > >> On a side note.. I dont suppose you have any idea if I can >> allocate some memory in a jGNEfilter? I dont mind stealing > >Since jGNEFilters are called from WNE/GNE and both of those traps are >documented as moving or purging memory, Yes you can allocate memory from >inside a jGNEFilter. If you want the memory to last longer than one pass >through your filter make sure to allocate the memory in the system heap, >otherwise it will be lost if the app quits. Or, you could receive consecutive calls from two different applications. Best, in any case, to allocate from the system heap if you want persistance. >If the memory will be allocated and disposed within one pass of the filter >it can go in the application's heap. Or maybe temporary memory if you don't want to use the app's heap. --Chris _______________________________________________________________________________ Chris Larson -- Amateur Macintosh Geek, CoBase Research Assistant L.A. Institute of Slowly and Painfully Working Out the Surprisingly Obvious Death to the Trojans! Go Bruins! (Insert disclaimer here) Internet: larson@kingston.cs.ucla.edu +++++++++++++++++++++++++++ >From gurgle@dnai.com (Pete Gontier) Date: Fri, 19 Aug 1994 09:45:18 -0800 Organization: Integer Poet Software In article <1994Aug19.161535.10734@cs.ucla.edu>, larson@oahu.cs.ucla.edu (Christopher Larson) wrote: > If an app installs a jGNEFilter, > is that filter visible to all other applications, or is it more like trap > patches (which, if done from an app, are only visible to the installing app)? It's visible to all other applications. -- Pete Gontier // CTO, Integer Poet Software // gurgle@dnai.com "Even during a particularly harsh (Colorado) winter... many of the 300 families in the VCTV (movies-on-demand) trial continued to go to video stores." -- eis@murrow.tufts.edu, in Wired 2.09 p62 +++++++++++++++++++++++++++ >From Jaeger@fquest.com (Brian Stern) Date: 19 Aug 1994 16:43:20 GMT Organization: The University of Texas at Austin, Austin, Texas In article <jedavis.777289328@Xenon.Stanford.EDU>, jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: > Jaeger@fquest.com (Brian Stern) writes: > > >In order to function properly any jGNEFilter has to insert itself into the > >chain of all jGNEFilters. When your filter exits it needs to jump to the > >previous filter. The address of the previous filter must be saved at the > >time that you install your filter. This saved address must be a global. > > Global? Well not really. Its not really good style, but Ive seen it done > by stuffing the address into your code at install time. The same way > many people store their app A5, with code that self modifies to > contain the location. > For that matter, where should I store my apps A5, so I can reset > it without using self modifying code? > > -James App? I thought we were talking about about an INIT. Installing a jGNEFilter from an app has a number of problems to worry about. I would probably write the filter as a code resource using A4 addressing. To install it I'd load the resource into the system heap, set up A0, and jump to it from the app to initilize it. It would save A5 for later use. It would also have to make sure that the app still existed. The A4 mechanism as used by Think C is also a self-modifying mechanism and I don't see any clean way around it. Having said that, I recently came across some code in a package called Extension Shell by Dair Grant that uses A4 addressing without being self-modifying. Here's the code: WRITTEN BY: Eric Shapiro (from July 1993 BYTE article, and the supporting source code) // Saves the current value of A4 on the stack, and puts the address of main // into A4. Should be called at start of stand-alone code resource to access globals. #define GetGlobals() asm { \ move.l A4, -(SP) ; save old A4 \ lea main, A4 ; get globals \ } // The converse of GetGlobals(). Restore the value of A4 #define UngetGlobals() asm { \ move.l (SP)+, A4 ; restore A4 from stack \ } This is dependent on main being at the beginning of your code resource and on your compiler compiling the code resource using A4 addressing. -- Brian Stern :-{)} Jaeger@fquest.com +++++++++++++++++++++++++++ >From bjaques@vanbc.wimsey.com (Barton Jaques) Date: Fri, 19 Aug 1994 09:36:35 -0700 Organization: Wimsey Information Services I spent many long hours slaving over my hot jGNEFilter before it worked correctly. It is the sloppiest hack I've seen in a while, but it works great. In short, here's what I do: ¥ I have a separate jGNEFilter.¹ project which compiles into a single resource. ¥ My INIT or DRVR reads that resource, detaches it, locks it. ¥ At startup time, my INIT or DRVR reads the current value of jGNEFilter, pokes it into my patch resource, then sticks the address of my patch resource into jGNEFilter. ¥ I have another handle which contains useful "global" variables and pointers to procedures contained in other code resources. I poke the handle into my detached jGNE resource. This way I have access to "globals" which I have scattered through different code resources É yet I don't have to fiddle with A5. ¥ The way I do this is by initialising a couple of local variables in my patch resource as follows: globRecH globalsH = 0xEEEEEEEE; long nextGNEFilter = 0xDDDDDDDD; My INIT or DRVR scans the resource, finds 0xEEEEEEEE, and overwrites it with my "globals" handle. Then it scans for 0xDDDDDDDD, and overwrites it with the pointer to the next GNE filter. ¥ If my GNEFilter calls a procedure in another of my code resources, THEN I need to set up A4/A5 as appropriate, and put it back before I return to my patch. ¥ When I want to de-install my driver, etc., I cauterise my patch by plugging 0x00000000 where my globals handle was. The filter then knows that it has been deactivated, so it passes to the next filter without doing anything. I can always restore the patch simply by poking globH back in again. This is the code in THINK C 68K. It surely will not work under another environment without serious modification, but it should illustrate the idea. void main() { globRecH globH = (globRecH)0xEEEEEEEE; long nextGNEFilter = 0xDDDDDDDD; EventRecord *filteredEventP; // declare any other local vars you need Ptr stackP; // stackP MUST be last local var declared // move.l (a7), -(a7) ; In actual res this line must precede LINK A6 // ; In ResEdit, insert '2F17' as 1st word in res // ; then move 2nd-last byte (= 4E5E) to // ; precede 5th-last byte (= 4A97) asm{ movem.l a7, stackP ; make copy of stack pointer movem.l a0-a3/d0-d7, -(sp) ; save registers } // if nextGNEFilter exists we must fiddle with stack to insert it if (nextGNEFilter) { stackP += ((Ptr)&globH - (Ptr)&stackP) + 8; *(long*)stackP = nextGNEFilter; } // do nothing if no globals if (globH) { // get a pointer to the event record asm{ move.l a1, filteredEventP ; test a1 (EventPtr) } eventwhat = filteredEventP->what; // now we can do our stuff } // restore registers and stack asm{ movem.l (sp)+, d0-d7/a0-a3 ; restore all registers tst.l (sp) ; in actual res these lines follow UNLK A6 (see note above) bne @END ; tst.l (sp)+ ; @END } // control now passes to next GNEFilter handler if any } -- bjaques@vanbc.wimsey.com +++++++++++++++++++++++++++ >From gurgle@dnai.com (Pete Gontier) Date: Fri, 19 Aug 1994 12:20:28 -0800 Organization: Integer Poet Software In article <bjaques-190894093636@pme22.pomo.wis.net>, bjaques@vanbc.wimsey.com (Barton Jaques) wrote: > I spent many long hours slaving over my hot jGNEFilter before it worked > correctly. It is the sloppiest hack I've seen in a while, but it works > great. In short, here's what I do... With all due respect for your long hours, quite frankly this all seems very convoluted and mildly dangerous. If any step in the development process goes wrong, the program fails at runtime. There are a minimum of helpful compile-time checks, and it's difficult to see why any one part of it works the way it does without understanding the whole picture. I'm glad it works for you, but it seems a lot simpler and a lot cleaner to simply use a strategy like that illustrated in "jGNE Helper" or whatever the hell that monthly posting is called. (It may sound like I didn't write it because I can't even remember its name, but I did write it, and you should assume I am accordingly biased.) -- Pete Gontier // CTO, Integer Poet Software // gurgle@dnai.com "Even during a particularly harsh (Colorado) winter... many of the 300 families in the VCTV (movies-on-demand) trial continued to go to video stores." -- eis@murrow.tufts.edu, in Wired 2.09 p62 +++++++++++++++++++++++++++ >From Jaeger@fquest.com (Brian Stern) Date: 20 Aug 1994 00:25:40 GMT Organization: The University of Texas at Austin, Austin, Texas In article <bjaques-190894093636@pme22.pomo.wis.net>, bjaques@vanbc.wimsey.com (Barton Jaques) wrote: > I spent many long hours slaving over my hot jGNEFilter before it worked > correctly. It is the sloppiest hack I've seen in a while, but it works > great. In short, here's what I do: Or, you could simply use A4 addressing to access your globals from your Filter. You can easily set a Boolean flag to let your Filter know if it's supposed to do anything. I don't trust any code building method that has instructions in it like ' In ResEdit, insert '2F17' as 1st word in res then move 2nd-last byte (= 4E5E) to precede 5th-last byte (= 4A97)'. In programming like in most things simpler is usually better. There are lot's simpler ways to do it than this. > -- > bjaques@vanbc.wimsey.com -- Brian Stern :-{)} Jaeger@fquest.com +++++++++++++++++++++++++++ >From jedavis@Xenon.Stanford.EDU (James Edward Davis) Date: 20 Aug 94 17:24:09 GMT Organization: Stanford University: Computer Science Department larson@oahu.cs.ucla.edu (Christopher Larson) writes: >In article <jedavis.777289328@Xenon.Stanford.EDU> jedavis@Xenon.Stanford.EDU (James Edward Davis) writes: >>Jaeger@fquest.com (Brian Stern) writes: >Well, you could use the above strategy, or if that is still to self-modify- >ish, you could install a Gestalt selector. Note that trickery to recover your >A5 is only necessary if your code might run when your app is switched out or >at interrupt time. If this is not the case, you can recover it from CurrentA5, >as you suggested earlier. >Hmmm. This raises an interesting question. If an app installs a jGNEFilter, >is that filter visible to all other applications, or is it more like trap >patches (which, if done from an app, are only visible to the installing app)? >Anyone? The jGNE filter *is* active across all apps. The low memory global that points to the filter chain, is not changed by context switches. This means you have to be very careful if you install a filter with code in your app heap, since if you quit without cleanup, theres a dangling reference and the machine will crash. -James +++++++++++++++++++++++++++ >From kluev@jonathan.srcc.msu.su (Kluev) Date: Mon, 22 Aug 94 23:02:46 +0400 Organization: (none) In article <jedavis.777403449@Xenon.Stanford.EDU> jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: > The jGNE filter *is* active across all apps. The low memory global that > points to the filter chain, is not changed by context switches. This > means you have to be very careful if you install a filter with code > in your app heap, since if you quit without cleanup, theres a dangling > reference and the machine will crash. One should be careful about this cleanup. One shouldn't restore original value in jGNEFilter, because someone may install gne filter on top of one's. Restoring jGNEFilter to original value will remove someone's filter and may crash him. The better solution is to install one's filter into system heap, never remove it, check if filter was already installed (Gestalt is one solution) (not to install it twice), set/clear flag on application's startup/exit to turn filter's activity on/off. Michael Kluev. +++++++++++++++++++++++++++ >From jedavis@Xenon.Stanford.EDU (James Edward Davis) Date: 23 Aug 94 02:50:02 GMT Organization: Stanford University: Computer Science Department kluev@jonathan.srcc.msu.su (Kluev) writes: >In article <jedavis.777403449@Xenon.Stanford.EDU> >jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: >One should be careful about this cleanup. One shouldn't restore >original value in jGNEFilter, because someone may install gne >filter on top of one's. Restoring jGNEFilter to original value >will remove someone's filter and may crash him. The better >solution is to install one's filter into system heap, never >remove it, check if filter was already installed (Gestalt is >one solution) (not to install it twice), set/clear flag on >application's startup/exit to turn filter's activity on/off. Yes, that would work but Id rather not leave memory allocated in the system heap. But your right about another being installed over mine. I allocate a small block in the system heap that just jumps to my code in the application heap. When I want to deinstall, I check to see if the front jGNEfilter is me. If it is, I can just reset the pointer. If its not, I change my stub to jump to the original address instead of my code. I only end up wasting a little system heap that way. I cant claim credit for the technique. I got a snippet from someone else, thought it was clever, and used it. -James +++++++++++++++++++++++++++ >From h+@nada.kth.se (Jon W{tte) Date: Tue, 23 Aug 1994 09:49:02 +0200 Organization: Royal Institute of Something or other In article <jedavis.777610202@Xenon.Stanford.EDU>, jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: >Yes, that would work but Id rather not leave memory allocated >in the system heap. But your right about another being installed >over mine. I allocate a small block in the system heap that just >jumps to my code in the application heap. When I want to deinstall, I hope you remember to flush the cache when you change this value... Cheers, / h+ -- Jon Wätte (h+@nada.kth.se), Hagagatan 1, 113 48 Stockholm, Sweden "Psychotherapist" - "Phycho-The-Rapist" Pure coincidence? You decide! +++++++++++++++++++++++++++ >From jedavis@Xenon.Stanford.EDU (James Edward Davis) Date: 23 Aug 94 22:00:51 GMT Organization: Stanford University: Computer Science Department h+@nada.kth.se (Jon W{tte) writes: >In article <jedavis.777610202@Xenon.Stanford.EDU>, >jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: >>Yes, that would work but Id rather not leave memory allocated >>in the system heap. But your right about another being installed >>over mine. I allocate a small block in the system heap that just >>jumps to my code in the application heap. When I want to deinstall, >I hope you remember to flush the cache when you change this >value... Why.. thats an excellent point. I dont flush it, but it seems to work fine on my C610. I think Ive been very lucky.. Ill go change that. Unless anyone can think of a legit reason its working. Thanks, James +++++++++++++++++++++++++++ >From kluev@jonathan.srcc.msu.su (Kluev) Date: Wed, 24 Aug 94 21:13:06 +0400 Organization: (none) In article <jedavis.777679251@Xenon.Stanford.EDU> jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: > h+@nada.kth.se (Jon W{tte) writes: > > >In article <jedavis.777610202@Xenon.Stanford.EDU>, > >jedavis@Xenon.Stanford.EDU (James Edward Davis) wrote: > > >>Yes, that would work but Id rather not leave memory allocated > >>in the system heap. But your right about another being installed > >>over mine. I allocate a small block in the system heap that just > >>jumps to my code in the application heap. When I want to deinstall, > > >I hope you remember to flush the cache when you change this > >value... > > Why.. thats an excellent point. I dont flush it, but it seems to > work fine on my C610. I think Ive been very lucky.. Ill go change > that. Unless anyone can think of a legit reason its working. This depends on processor type and what your code is looked like. If it is looked like that: JMP Address; 0x4EF9 0x12345678; where 0x12345678 is modified address, you need to flush cache on 040 processors. If it is looked like this: MOVE.L *+$0006, -(sp); 0x2F3A0004; RTS; 0x4E75 DC.L 0x12345678; 0x12345678; where 0x12345678 is modified addr, don't worry about flushing cache. (DC.L will not run as code.) /* From MacTech magazine */ Michael Kluev. --------------------------- End of C.S.M.P. Digest **********************