TweetFollow Us on Twitter

Apr 86 Tech Questions
Volume Number:2
Issue Number:4
Column Tag:Ask Professor Mac

Reader's Technical Questions

By Steve Brecher, Software Supply, 4618 E. Sixth St., Long Beach, CA. 90814

Steve only writes when someone sends him an interesting question to answer. Help us keep Steve busy by sending in your technical questions to Professor Mac. You may write to Steve directly at his software store listed above.

Desk Accessories and JIODone

Q. Jan Eugenides sent me a question about how the Control routine of a desk accessory should return: via an RTS instruction or via a JMP through JIODone? This turns out to be a somewhat complex issue; the following discussion is based on answers from my most trusted authority, the ROM, and insights provided by Lew Rollins and Jon Hueras via the CompuServe Mac Developers SIG.

A. Inside Macintosh says that the Control routine of a driver should return to the ROM's IODone routine -- to the address contained in the system global variable JIODone. When jumping through JIODone, register A1 must contain the address of the driver's Device Control Entry (DCE); otherwise, no registers need to be preserved by the Control routine. Generally, any queued request to a driver must return to IODone so that the system won't hang waiting for completion of the request.

If a driver doesn't need to be locked in the heap when it is not executing, it has the dRAMBased bit in the DCE set and the dNeedLock bit clear. Note that a driver is always locked before it is entered, so self-locking by a driver is superfluous. The dNeedLock bit specifies whether the driver needs to be locked when it is not executing. If dNeedLock is clear, IODone will unlock the driver and the DCE.

For queued I/O requests, IODone removes the completed request from the queue and dequeues the next pending request, if any, for execution. However, Desk Accessory (DA) Control requests are not queued (they have the noQueue bit set in the Conrol trap word) except for the case of the "goodbye kiss" (csCode = -1) call that is issued if the dNeedGoodbye bit is set in the driver header.

But... the IODone routine in the 64K ROM has a bug (fixed in the 128K ROM). In order to determine if the request was a queued one, it looks at the noQueue (,IMMED) bit in the trap word in the queue element pointed to by the dCtlQueue field of the DCE. Problem is, if the request was not a queued one, then it's not in the queue to be examined! Since DA Control calls (except for a "goodbye kiss") are not queued, IODone will be using a Nil pointer in the dCtlQueue field and will examine the word at offset ioTrap (offset 6) from address zero, i.e., address 00000006. As luck would have it, usually the noQueue bit happens to be set at that irrelevant location, and IODone doesnt't try to remove the nonexistent queue element - so the bug doesn't usually bite. But it's dicey for a DA to depend on that happenstance.

A further complication for some DAs is that their Control routines may be reentrant. Consider the case of a DA that has the dNeedTime bit set in its header because it wants to be called periodically. Suppose this DA's Control routine calls ModalDialog (possibly indirectly via SFGet/PutFile or Alert). ModalDialog calls SystemTask, which in turn may issue a periodic Control call to the DA. If the DA is always unlocked at Control exit (either explicitly by itself or by IODone), then the periodic call will at exit unlock the DA/DCE, and when ModalDialog returns the hapless DA will find itself "mysteriously" unlocked with perhaps disastrous results. If you share Jon Heuras's "sheer paranoia" about reentrancy, you can use his technique of clearing the dCtlEnable bit in the DCE at entrance to Control, and setting it again at exit -- this avoids reentrant Control calls.

Ok, so those are the facts -- what does it all mean? I would suggest the following approach. It assumes that the DA does not need itself nor its DCE to be locked while the DA is not executing; and it permits reentrance. There are few cases when a DA must be locked when not executing, and it is impolite to the host application to have unnecessarily-locked blocks in the heap.

Allocate a word -- or to be safe, a longword -- in the DAs private storage to be used as a Control entrance counter. Clear this counter in the Open routine. At entrance to the Control routine, increment the counter. Then the Control exit logic would look like this:

;A0 points to the request parameter block
;A1 points to the DCE
;the following 4 lines can be omitted if dNeedGoodbye is clear
;- otherwise we assume private storage has beed disposed of
 Btst #noQueueBit-8,ioTrap(A0) ;queued request?
 Beq.S  @0;no
 Move.L JIODone,A0 ;yes, it's "Goodbye"...
 Jmp  (A0);so return to IODone
@0 Move.L dCtlStorage(A1),A0;get storage handle
 Move.L (A0),A2  ;dereference
 SubQ.L #1,EntranceCount(A2);decrement counter
 Bne.S  @1;br if reentered
 _HUnlock ;unlock storage
 Move.L A1,A0  ;DCE pointer
 _RecoverHandle  ;DCE handle
 _HUnlock ;unlock DCE
 Lea  Driver,A0  ;addr of this DRVR
 _RecoverHandle  ;handle to ourself
 _HUnlock ;unlock ourself
@1 Rts  ;bypass IODone

Scamble Well, Please

Q. Dr. David T. Linker of Trondheim, Norway, writes, “I was encouraged by your comment on 'dumb' questions [I solicited them! --SB]. Why do Macintosh debuggers offer to scramble the heap? I use the heap ... but I can't think of a reason why I would want to scramble it. It seems that this would be undesireable, kind of like 'scramble variables.'"

A. "Rearrange" would be a more accurate word for the function offered by the debuggers. Scrambling the heap means to move relocateable blocks around; if your code at some point depends on the incorrect assumption that a given block is locked (won't move), then scrambling the heap may help to reveal that bug. After the scramble, the program won't find what it expects to find at what it thinks is the address of the block, and will probably behave differently than it would without the scrambling.

Such a buggy program may have seemed to be healthy only because, by luck, the block has not moved; but in some circumstances that testing has not yet encountered, the block will move, and the program will crash or otherwise misbehave. If your program can survive a heap scramble on any trap which might rearrange the heap, then you can have some confidence that it is not making incorrect assumptions about blocks being immobile.

Application Icon

Q. This and the following question were submitted by L. Tannenbaum of Long Beach, Calif. "How do I get the Finder to use the 'generic' icon for an application? What info do I include or exclude in the resource file?"

A. This is an unusual question - most often, people want to know how to give their application a custom icon. To get the generic "hand-on-diamond" icon on the Finder desktop, all you need do is specify APPL for the type of the application file. Begin your RMaker input file with:

Name Of My Application File
APPLMYAP

This examples uses MYAP as the signature of the application.

Moving Pictures

Q. How do I get a "PICT," say one I drew in MacPaint, into a resource file? I have RMaker (with no documentation) and REdit (not ResEdit).

A. Use RMaker to create a dummy PICT resource in the target resource file:

TYPE PICT = GNRL ;; define a "new" type via GNRL

,128 ;; resource ID

.I ;; integers follow

0 0 0 0 ;; dummy rectangle coordinates

In MacPaint, cut your picture and paste it into the Scrapbook. Now run REdit and open your resource file and the PICT resource type. You'll see PICT 128 represented by an icon. Open the Scrapbook, cut the picture from it, select the PICT 128 icon, and paste. (P.S. ResEdit is more powerful than REdit; ResEdit lets you create resources as well as modify them. ResEdit 1.0d5 is available on the MacTutor Utility disk and source code disk #6. ]

AutoWatch

It's a service to the user for an application to display a wristwatch cursor whenever the application is busy and not in the mood for user input. But it's something of a nuisance for the programmer to explicitly change the cursor before and after each time-consuming operation; besides, sometimes it's not easy to know a priori that an operation will be time-consuming.

I saw a message from Larry Rosenstein of Apple on one of the networks in which he outlined a scheme for automating changing of the cursor to a watch and back. The basic idea is to install both a vertical-blanking (VBL) interrupt task and a hook into GetNextEvent. If GetNextEvent has not been called for some pre-defined period of time, then the application is deemed to be "busy" and the cursor is changed to a watch; this is done by the VBL task.

The system's VBL interrupt handler checks each element in a VBL queue each 60th of a second, decrementing a counter value in the queue element. If the counter decrements to zero, a task associated with that VBL queue element is executed. Then -- unless the VBL task reinitialized the counter to a non-zero value -- the element is removed from the queue. For more information, see the Vertical Retrace Manager chapter of Inside Macintosh.

Suppose we want a watch cursor to be displayed whenever half a second passes with no calls to GetNextEvent. What we do is install a VBL task with a value of 30 (30/60 of a second) in the associated VBL queue element counter field. We also revector the GetNextEvent trap to a piece of code which restores the counter in the VBL queue element to 30 (and then does normal GetNextEvent processing). If GetNextEvent is called at least once every half second, the VBL task will never be executed, because the VBL queue element's counter will never get decremented to zero. But if half a second elapses with no calls to GetNextEvent, the VBL task will be executed.

When the VBL task is executed, it examines a global flag which indicates whether the cursor has already been changed to a watch. If the flag is false, the task saves the current cursor, changes the screen cursor to a watch, and sets the flag. Then it restores the counter field in the VBL queue element so that the element will not be dequeued.

The GetNextEvent hook code restores the VBL task's timer value, and, if the VBL task changed the cursor from a non-watch to a watch (indicated by the global flag set by the VBL task), restores the original cursor.

In simplest terms, every time GetNextEvent is called it says to the Vertical Retrace Manager, "Whoa! The application is not busy! Restart your countdown." The Vertical Retrace Manager, each 60th of a second, decrements the counter; if it has reached zero, that means GetNextEvent has not been called since the countdown last started, implying a busy application which needs a watch cursor: the VBL task is executed and changes the cursor.

There are a couple of minor complications. The GetNextEvent hook code cannot just do an InitCursor (which makes the cursor an arrow), because the pre-watch cursor may not have been an arrow. But neither can it blindly restore whatever cursor the VBL task saved. Consider this example: the application calls DIBadMount to initialize a disk. DIBadMount changes the cursor to a watch as it starts to format the disk. Shortly thereafter, the VBL task countdown reaches zero, and the VBL task saves the current cursor (a watch!) and sets the cursor to a watch - superfluous, but no harm done. When the disk format completes, DIBadMount does an InitCursor, and returns to the application. The application calls GetNextEvent. Now our GetNextEvent hook sees the flag set by the VBL task indicating that the VBL task saved the old cursor and changed the cursor to a watch; so GetNextEvent restores the old cursor. No good! We've restored to a watch when we don't want a watch. To avoid this, the GetNextEvent hook looks at the cursor saved by the VBL task; if it's a watch, it doesn't restore it.

Also, there are times when GetNextEvent is not called, but nonetheless it would be inappropriate to change the cursor to a watch: when the mouse is being tracked by, e.g., the Menu Manager or the Control Manager. If the user dawdles while he has a menu pulled down, we can't change the cursor just because GetNextEvent is not being called. So, the VBL task will not change the cursor if the mouse button is down.

The time period used to initialize the VBL counter is application-dependent. About half a second works pretty well with an application I recently implemented. If the period is too short, the cursor will too-often toggle to a watch and back in a distracting way. If the period is too long, then the user will not be promptly informed that the application is busy, and the change of the cursor to a watch will seem unrelated to the preceding user action which initiated the time-consuming process.

My implementation of AutoWatch in MDS Assembler is shown in Figure 1. This implementation will not work under Switcher.

Move Low

The 128K ROM has a new feature with regard to the loading of CODE segments. If a CODE resource does not have the resLocked attribute, then the Segment Loader will load that segment as high as possible in the heap. The idea is that up at or near the top of the heap, it will be "out of the way" and not contribute to heap fragmentation.

However, this can be a problem with CODE 1, the code segment which is loaded first and which is usually the main segment of the application. Upon loading, the Segment Loader will lock it, regardless of the resLocked attribute, as it does for all segments. It will remain locked until the applicaton unloads it - but most applications don't unload CODE 1. If CODE 1 does not have the resLocked attribute (which it will not if generated by, e.g., the Consulair Linker), it will be loaded high in the heap - the initial heap before any heap expansion has occurred. Since almost all applications expand the heap either explicitly or implicitly, the new Segment Loader feature is effectively a "load in middle" for CODE 1.

To get around this, I implemented a routine (Figure 2) which moves the calling segment to or near the bottom of the heap. It consists of two parts: the invoking code, and a subroutine which does the actual move. The invoking code puts the subroutine on the stack and calls it; the subroutine reallocates the calling segment's heap block low in the heap, displaces its return address by the distance the segment was moved, and returns to the caller, which cleans up the stack. I put this code right after a call to MaxApplZone at the beginning of my CODE 1 segment.

The reason the subroutine is executed from the stack is that the Memory Manager call which it uses to get space low in the heap, _ReservMem, may move the original segment which at that point is unlocked. If it does move, and _ReservMem were called from within the moved segment, then _ReservMem would return to the wrong place.

You may notice that after _ReservMem, the original heap block containing the segment has been released - it's now a free block. Nonetheless we copy the segment contents from it to the new location low in the heap. That's OK - nothing can happen to clobber the freed block between the _ReservMem and the _BlockMove.

; Figure 1 Autowatch routine
;  by Steve Brecher, MacTutor 1986
; 
 XDEF AutoWatch,ShowWatch
;
; Thanks to Larry Rosenstein for the idea.
;
; This code must be in a  locked segment (e.g., CODE 1).
;
;
; Procedure AutoWatch(TickValue: integer);
;
; If TickValue <> 0, install VBL task that will change cursor
;to watch after TickValue ticks since GetNextEvent
;was called, provided that mouse button is not down. 
;Old cursor is saved before cursor is changed to watch.
;Installs GetNextEvent hook that restores saved cursor
;and reinits VBL countdown.
; Calls _InitCursor.
;Call AutoWatch with TickValue<>0 just before entering
;     your event loop.
; If TickValue = 0,  remove VBL task and GetNextEvent hook. 
;Call before quitting application.
;
; Procedure ShowWatch;
;Unconditionally change cursor to watch. (Must
;have called AutoWatch with TickValue <> 0 at
;some time previous.)  Typically called right before
;exiting application, to cover time until Finder (or
;  whatever is next) comes up.
;
 IncludeMacTraps.D
 IncludeSysEqu.D
 IncludeQuickEqu.D

 IncludeMacros ;see MacTutor Jan. 1986 ie,
 ; StackFrame,  Arg, Result, Local, Return
 ; Pop ..., Push ...,  Assume
;
;  Global variables
;
GNEptr  DS.L1
GNEhookPtrDS.L 1
GNEreturn DS.L 1
VBLTicksDS1
watchHndl DS.L 1
VBLCountPtr DS.L 1
SavedCursor DS.B cursRec
isWatch DS.B1
;
; The VBL queue element.
;
VBLQElem:
 DC.L 0 ;link
 DCvType
 DC.L 0 ;task ptr
 DC0  ;countdown value
 DC0  ;phase
;
; The VBL task code.
;
VBLtask:
 Move.L CurrentA5,A2 ;application's A5 into A2
 Tst.B  MBState  ;mouse button down?
 Bpl.S  @1;yes, don't change cursor
 Bset #0,isWatch(A2)   ;already make it a watch?
 Bne.S  @1;yes
 Lea  TheCrsr,A0 ;no, save current cursor...
 Lea  SavedCursor(A2),A1  
 Assume CursRec&3 = 0
 MoveQ  #(CursRec/4)-1,D0
@0 Move.L (A0)+,(A1)+
 Dbra D0,@0
 Move.L watchHndl(A2),A0  ;change to watch...
 Push.L (A0)
 _SetCursor
@1 Lea  VBLQElem+vblCount,A0  ;re-init VBL count
 Move VBLticks(A2),(A0)
 Rts
;
; The GetNextEvent hook code.  It's moved to system heap,
; which is why we must use a global to store the VBL queue
;  element count field address -- the Lea instruction can't be 
; used in code that's moved relative to the target of the Lea.
;
GNEhook:
 Pop.L  GNEreturn(A5)
 Bclr #0,isWatch(A5) ;did we make it a watch?
 Beq.S  @1;no
 Move.L watchHndl(A5),A0  ;yes, but did VBL task         ;"change" watch 
to watch?  (maybe
 ; somebody made it a watch behind our back)
 Move.L (A0),A0
 Lea  SavedCursor(A5),A1
 MoveQ  #(CursRec/4)-1,D0
@0 CmpM.L (A0)+,(A1)+;compare cursor saved by
 ; VBL task to watch...
 Dbne D0,@0
 Beq.S  @1;don't restore if cursor is a watch
 Pea  SavedCursor(A5);ok, restore non-watch 
 _SetCursor
@1 Move.L GNEptr(A5),A0 ;addr of original GetNextEvent 
 Jsr  (A0);go do GetNextEvent
 Move.L VBLCountPtr(A5),A0 ;re-init VBL countdown...
 Move VBLticks(A5),(A0)
 Move.L GNEreturn(A5),A0
 Jmp  (A0);return to trap dispatcher
GNEhookLen     Equ *-GNEhook

 StackFrame NotLinked
 Arg    TickValue,word

AutoWatch:
 Move TickValue(SP),VBLTicks(A5) ;install or remove?
 Beq.S  Remove   ;remove
 _InitCursor
 SFIsWatch(A5)
 Lea  VBLQElem+vblCount,A0 ;store addr of VBL
 Move.L A0,VBLCountPtr(A5)   ; queue elem count field
 Move VBLticks(A5),(A0)   ; for use by GetNextEvent
 ; hook.  Init VBL countdown 
 ; value in queue element.
 MoveQ  #CursRec,D0;length of a cursor
 _ResrvMem;make space low in heap
 Push.L ;room for _GetResource result
 Move.L #'CURS',-(SP)
 Push #watchCursor
 _GetResource
 Pop.L  A0
 Move.L A0,watchHndl(A5)  ;save handle to watch 
 _HNoPurge;don't let it go away
 Lea  VBLQElem,A0
 Lea  VBLtask,A1
 Move.L A1,VBLaddr(A0)
 _VInstall;install the VBL task
 MoveQ  #GNEhookLen,D1  ;length of our hook code
 Move.L D1,D0
 _NewPtr,SYS;move hook code to system heap...
 Move.L A0,A1
 Move.L A1,GNEhookPtr(A5) ;save addr in system heap 
 ; for remove
 Lea  GNEhook,A0
 Move.L D1,D0
 _BlockMove
 Move #$170,D0   ;GetNextEvent trap#
 _GetTrapAddress
 Move.L A0,GNEptr(A5);save addr of original code 
 Move.L A1,A0    ;addr of our hook code
 Bra.S  SetGNE   ;revector GetNextEvent to our 
 ; hook
Remove:
 Lea  VBLQElem,A0
 _VRemove ;remove the VBL queue element
 Move.L GNEhookPtr(A5),A0
 _DisposPtr ;dispose of hook code in sys heap
 Move.L GNEptr(A5),A0;addr of original GetNextEvent 
SetGNE: Move#$170,D0
 _SetTrapAddress ;set GetNextEvent routine address
 Return

ShowWatch:
 Move.L watchHndl(A5),A0
 Push.L (A0)
 _SetCursor
 Rts

 End
Figure 2
; 
SegStart: ;first location in segment (past segment header)

 ;(any data or miscellaneous stuff)
;
;  MoveLo subroutine.  This subroutine is moved to the stack and executed 
from there.
;
;  Unlock the calling segment, reallocate it low in the heap, copy its 
contents,
;  lock the calling segment, and return to the calling segment.
; 
;  On entry, A0 = handle to calling segment.
;
MoveLo: Pop.L  D2;return address
 Sub.L  (A0),D2  ;D2.W = relative return address
 Push.B (A0);save Memory Mgr flags in master ptr
 Clr.B  (A0);make sure not purgeable, unlocked
 _GetHandleSize  ;get his size
 Move.L D0,D1  ;save size
 _ResrvMem;make space low in the heap 
 ; (may move caller)
 Push.L (A0);save addr of calling segment
 Move.L D1,D0  ;size
 _ReallocHandle  ;reallocate handle in low space
 Move.L A0,D0  ;save handle
 Move.L (A0),A1  ;dest addr
 Pop.L  A0;source addr for move
 Exg  D1,D0 ;put size in D0, saved handle in D1
 _BlockMove ;copy contents of segment
 Move.L D1,A0  ;handle
 Pop.B  (A0);restore flags in master pointer
 Jmp  (A1,D2)  ;return
MoveLoSizeEqu  *-MoveLo

 XDEF Start ;application entry point
Start:  Bsr MaxApplZone 
 ;see "Ask Prof. Mac" Aug. 1985 
; Move this CODE resource down in heap
;
 MoveQ  #MoveLoSize,D0  ;size of MoveLo routine
 Sub  D0,SP ;make room on stack for routine
 Lea  MoveLo,A0  ;source addr
 Move.L SP,A1    ;dest addr
 _BlockMove ;move MoveLo routine to stack
 Lea  SegStart-4,A0;addr of segment header
 _RecoverHandle  ;pass segment handle to MoveLo
 Jsr  (A1);call MoveLo on stack
 Add  #MoveLoSize,SP ;pop routine from stack
;
; The usual initiation rites:
;(call _MoreMasters here as many times as needed)
 MoveQ  #0,D0
 SubQ #1,D0 ;all-events mask
 _FlushEvents
 Pea  -4(A5)
 _InitGraf
 _InitFonts
 _InitWindows
 _TEInit
 Push0.L
 _InitDialogs

Notes from Volume 2 Number 5:

Ask Prof. Mac Correction for April

Steve Brecher

Please note an error on page 62 in the April 1986 issue of MacTutor (volume 2 number 4) in the example of using JIODone. After testing the "noQueueBit", the next instruction says "Beq.S @0". This is incorrect. It should be "Bne.S @0" instead. If this bit is set, then the branch will be taken and the rest of the code implemented, which is what we want for a "no queued" item. For queued items, the branch will not be taken, and we will exit through JIODone.

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

f.lux 42.1 - Adjusts the color of your d...
f.lux makes the color of your computer's display adapt to the time of day, warm at night and like sunlight during the day. Ever notice how people texting at night have that eerie blue glow? Or wake... Read more
Spotify 1.1.94.872 - Stream music, creat...
Spotify is a streaming music service that gives you on-demand access to millions of songs. Whether you like driving rock, silky R&B, or grandiose classical music, Spotify's massive catalogue puts... Read more
Vitamin-R 4.15 - Personal productivity t...
Vitamin-R creates the optimal conditions for your brain to work at its best by structuring your work into short bursts of distraction-free, highly focused activity alternating with opportunities for... Read more
OfficeTime 2.0.628 - Easy time and expen...
OfficeTime is time and expense tracking that is easy, elegant and focused. Other time keepers are clumsy or oversimplified. OfficeTime balances features and ease of use, allowing you to easily track... Read more
Slack 4.28.182 - Collaborative communica...
Slack brings team communication and collaboration into one place so you can get more work done, whether you belong to a large enterprise or a small business. Check off your to-do list and move your... Read more
DEVONthink Pro 3.8.6 - Knowledge base, i...
DEVONthink is DEVONtechnologies' document and information management solution. It supports a large variety of file formats and stores them in a database enhanced by artificial intelligence (AI). Many... Read more
FileMaker Pro 19.5.4 - Quickly build cus...
FileMaker Pro is the tool you use to create a custom app. You also use FileMaker Pro to access your app on a computer. Start by importing data from a spreadsheet or using a built-in Starter app to... Read more
Backblaze 8.5.0.628 - Online backup serv...
Backblaze is an online backup service designed from the ground-up for the Mac. With unlimited storage available for $6 per month, as well as a free 15-day trial, peace of mind is within reach with... Read more
Day One 7.16 - Maintain a daily journal.
Day One is an easy, great-looking way to use a journal / diary / text-logging application. Day One is well designed and extremely focused to encourage you to write more through quick Menu Bar entry,... Read more
Garmin Express 7.14.0.0 - Manage your Ga...
Garmin Express is your essential tool for managing your Garmin devices. Update maps, golf courses and device software. You can even register your device. Update maps Update software Register your... Read more

Latest Forum Discussions

See All

We’re Digging ‘Shovel Knight Dig’ – The...
We spend the bulk of this week’s podcast talking about the new iPhone 14. Specifically, the iPhone 14 Pro Max which both Eli and myself picked up. The consensus seems to be: They’re great! They’re iPhones! We do lay down our hot takes on all the new... | Read more »
TouchArcade Game of the Week: ‘Loose Noz...
There aren’t a lot of stories like that of the development of Loose Nozzles, and of those games that do have an interesting development story, even fewer are actually decent games to play. Loose Nozzles nails both, though. The way it was created is... | Read more »
SwitchArcade Round-Up: ‘Shovel Knight Di...
Hello gentle readers, and welcome to the SwitchArcade Round-Up for September 23rd, 2022. In today’s article, we’ve got the rest of this week’s releases to look at. There are actually a few big games today, including the hot-hot-hot Shovel Knight Dig... | Read more »
‘Gubbins’ is a Way Too Adorable Word Gam...
There are games whose art style, sounds, and overall vibe just make me smile ear to ear. Games like Hidden Folks, Krispee Street, or Tiny Wings. There’s just something so cool about being able to literally feel the heart that goes into a game. Now... | Read more »
Based on the Baking Reality Show, ‘Naile...
Fans of Netflix’s reality baking show Nailed It! have a new holiday-themed season to look forward to next month when Nailed It! Halloween launches on October 5th, but the fun doesn’t stop there because the show is also arriving as a mobile game the... | Read more »
Cookie Run: Kingdom announces collaborat...
In news sure to excite fans of biscuits or K-Pop music, the Korean sensations BTS have teamed up with Cookie Run: Kingdom for a series of events. After some warm-up episodes, the collaboration will culminate in a BTS in-game concert, so if anyone'... | Read more »
‘Shovel Knight Dig’ From Nitrome and Yac...
Shovel Knight Dig () from Nitrome and Yacht Club Games is this week’s new Apple Arcade release. It is definitely one of my favorite additions to the service ever, and a fantastic game overall. I played it a few hours ago when it started rolling out... | Read more »
SwitchArcade Round-Up: ‘Mario Strikers’...
Hello gentle readers, and welcome to the SwitchArcade Round-Up for September 22nd, 2022. Hunh, lots of twos in the date today. Nifty. As those who read yesterday’s article may remember, I got a vaccine shot about twenty four hours ago and it is... | Read more »
Rogue-Like Platformer ‘Tallowmere 2’ Lau...
The original Tallowmere from developer Chris McFarland launched on mobile way back in 2015, and to be honest it did not leave a good first impression with me. For lack of a better term, it just seemed… janky, and right from the start the game sort... | Read more »
Alchemy Stars newest event launches and...
Alchemy Stars has introduced its latest event, entitled Farewell, My Wonderland, bringing with it new characters and a bevvy of rewards. The event will reportedly focus on the underlying message that even after tragic events there is still light,... | Read more »

Price Scanner via MacPrices.net

Use our exclusive Apple Price Trackers to fin...
Our Apple award-winning price trackers are the best place to look for the lowest prices and latest sales on all the latest Apple gear this season. Scan our price trackers for the latest information... Read more
New promo at Verizon: Get Apple Watch Series...
Purchase a new iPhone 14 at Verizon, and get an Apple Watch Series 8 for as low as $5 per month. $120 in promo credits for the Watch are spread over a 36 month term, reducing the price of the Watch... Read more
Visible drops prices on Apple iPhone 13 model...
Verizon’s low-cost wireless cell service, Visible has dropped prices on iPhone 13 models to new low prices starting at $599: – iPhone 13 Pro Max: starting at $980 + free $200 gift card – iPhone 13... Read more
Back in stock! 14″ MacBook Pros with Apple M1...
Amazon has restocked 14″ MacBook Pros M1 Pro CPUs for $400 off MSRP, starting at only $1599. Shipping is free. Be sure to make your purchase from Amazon rather than a third-party seller. Their prices... Read more
This is the final week to take advantage of A...
Apple’s Back to School promotion for 2022 ends on September 26, 2022. As part of this promotion, Apple will include a free $150 Apple Gift Card with the purchase of any MacBook Air, MacBook Pro, or... Read more
Mac Studio with M1 Max CPU back in stock toda...
Apple has the base standard-configuration Mac Studio available again in their Certified Refurbished section for $1799, and it’s in stock today. Each Mac Studio comes with Apple’s one-year warranty,... Read more
Apple MagSafe iPhone battery on sale for $84,...
Amazon has Apple’s MagSafe Battery on sale for $84 today. Shipping is free. That’s $15 off Apple’s MSRP, and it’s the lowest price for one of these MagSafe batteries among the Apple retailers we... Read more
24-inch M1-powered iMacs available today at A...
Apple has a full range of 24-inch M1 iMacs available today in their Certified Refurbished store. Models are available starting at only $1099 and range up to $260 off original MSRP. Each iMac is in... Read more
Verizon offers free Apple iPhone 14 models to...
Verizon is offering a $800-$1000 discounts on Apple’s new iPhone 14 models for new and existing customers with a qualified trade-in. Price of the iPhone 14 will be spread over 36 months of payments,... Read more
Gazelle drops prices on iPhone 13 models to a...
Gazelle has a full line of discounted, refurbished, unlocked Apple iPhone 13 models now available starting at $469. iPhones are offered in Fair, Good, and Excellent conditions, and multiple colors... Read more

Jobs Board

Physician Assistant, Primary Care, *Apple*...
Physician Assistant, Primary Care, Apple Valley (1.07FTE) + Job ID: 65766 + Department: AV Primary Care + City: Apple Valley, MN + Location: HP - Apple Read more
Operations Manager - Mac/ *Apple* Engineerin...
…Responsible for the day-to-day activities relating to the engineering of Apple Macs in a complex, multi-platform environment. Demonstrates strong leadership, Read more
Lead Developer - *Apple* tvOS - Rumble (Uni...
…earnings, and positive sentiment About the role: We are looking for a Lead Apple tvOS Developer to join our application engineering team to expand our video centric Read more
Systems Administrator - *Apple* Devices / J...
…Administration **Duties and Responsibilities** + Configure and maintain the client's Apple Device Management (ADM) solution. The current solution is JAMF supporting Read more
Sr Product Manager, *Apple* TV Platforms -...
…an experienced senior product manager to drive the strategy and requirements for our Apple TV devices, acting as the champion and owner of the holistic experience in Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.