/* -*-mode: C; fill-column: 78; c-basic-offset: 4; compile-command: "make ARCH=ARM_ONLY MEMDEBUG=TRUE"; -*- */ /* * Copyright 2004-2007 by Eric House (xwords@eehouse.org). All rights * reserved. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include #include #include #include #ifdef XWFEATURE_BLUETOOTH #include #include #endif #include "pace_man.h" #include "palmmain.h" /* for custom event types */ #include "palmsavg.h" #include "strutils.h" /* Looks like I still need this??? */ void* memcpy( void* dest, const void* src, unsigned long n ) { void* oldDest = dest; unsigned char* d = (unsigned char*)dest; unsigned char* s = (unsigned char*)src; FUNC_HEADER(memcpy); if ( dest < src ) { while ( n-- > 0 ) { *d++ = *s++; } } else { d += n; s += n; while ( n-- > 0 ) { *--d = *--s; } } FUNC_TAIL(memcpy); return oldDest; } /* memcpy */ unsigned long Byte_Swap32( unsigned long l ) { unsigned long result; result = ((l >> 24) & 0xFF); result |= ((l >> 8) & 0xFF00); result |= ((l << 8) & 0xFF0000); result |= ((l << 24) & 0xFF000000); return result; } unsigned short Byte_Swap16( unsigned short l ) { unsigned short result; result = ((l >> 8) & 0xFF); result |= ((l << 8) & 0xFF00); return result; } void write_unaligned32( unsigned char* dest, unsigned long val ) { int i; dest += sizeof(val); for ( i = 0; i < sizeof(val); ++i ) { *--dest = val & 0x000000FF; val >>= 8; } } /* write_unaligned32 */ void write_unaligned16( unsigned char* dest, unsigned short val ) { int i; dest += sizeof(val); for ( i = 0; i < sizeof(val); ++i ) { *--dest = val & 0x00FF; val >>= 8; } } /* write_unaligned16 */ #define write_unaligned8( p, v ) *(p) = v unsigned short read_unaligned16( const unsigned char* src ) { int i; unsigned short val = 0; for ( i = 0; i < sizeof(val); ++i ) { val <<= 8; val |= *src++; } return val; } /* read_unaligned16 */ #define read_unaligned8(cp) (*(cp)) unsigned long read_unaligned32( const unsigned char* src ) { int i; unsigned long val = 0; for ( i = 0; i < sizeof(val); ++i ) { val <<= 8; val |= *src++; } return val; } /* read_unaligned32 */ #ifdef XWFEATURE_FIVEWAY #include "pnostate.h" Err HsNavDrawFocusRing (FormType* formP, UInt16 objectID, Int16 extraInfo, RectangleType* rP, HsNavFocusRingStyleEnum ringStyle, Boolean forceRestore) { Err result; FUNC_HEADER(HsNavDrawFocusRing); RectangleType RectangleType_68K1; SWAP_RECTANGLETYPE_ARM_TO_68K( &RectangleType_68K1, rP ); { PNOState* sp = GET_CALLBACK_STATE(); STACK_START(unsigned char, stack, 18); ADD_TO_STACK2(stack, hsSelNavDrawFocusRing, 0); ADD_TO_STACK4(stack, formP, 2); ADD_TO_STACK2(stack, objectID, 6); ADD_TO_STACK2(stack, extraInfo, 8); ADD_TO_STACK4(stack, &RectangleType_68K1, 10); ADD_TO_STACK1(stack, ringStyle, 14); ADD_TO_STACK1(stack, forceRestore, 16); STACK_END(stack); result = (Err)(*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(sysTrapHsSelector), stack, 18 ); sp->emulStateP->regA[7] -= 2; } FUNC_TAIL(HsNavDrawFocusRing); return result; } #endif /* Need to parse the format string */ Int16 StrPrintF( Char* s, const Char* formatStr, ... ) { unsigned long* inArgs = ((unsigned long*)&formatStr) + 1; return StrVPrintF( s, formatStr, (_Palm_va_list)inArgs ); } /* StrPrintF */ /* from file StringMgr.h */ Int16 StrVPrintF( Char* s, const Char* formatStr, _Palm_va_list arg ) { Int16 result; unsigned long* argv_arm = (unsigned long*)arg; unsigned char argv_68k[48]; unsigned short done, isLong, innerDone, useArg; unsigned char* str = (unsigned char*)formatStr; unsigned short offset = 0; for ( done = 0; !done; ) { switch( *str++ ) { case '\0': done = 1; break; case '%': isLong = useArg = 0; for( innerDone = 0; !innerDone; ) { unsigned char nxt = *str++; switch( nxt ) { case '%': innerDone = 1; break; case 'l': isLong = 1; break; case 's': isLong = 1; case 'd': case 'x': innerDone = 1; useArg = 1; break; default: if ( nxt >= '0' && nxt <= '9' ) { /* accept %4x */ } else { crash(); } } } if ( useArg ) { unsigned long param; param = *argv_arm++; if ( isLong ) { write_unaligned32( &argv_68k[offset], param ); offset += 4; } else { write_unaligned16( &argv_68k[offset], (unsigned short)param ); offset += 2; } } break; } } /* now call the OS.... */ { PNOState* sp = GET_CALLBACK_STATE(); STACK_START(unsigned char, stack, 12); ADD_TO_STACK4(stack, s, 0); ADD_TO_STACK4(stack, formatStr, 4); ADD_TO_STACK4(stack, argv_68k, 8); STACK_END(stack); result = (Int16) (*sp->call68KFuncP)( sp->emulStateP, // NOT sysTrapStrPrintF !!!! PceNativeTrapNo(sysTrapStrVPrintF), stack, 12 ); } return result; } /* StrVPrintF */ /* Events. Need to translate back and forth since the ARM struct looks * different from the 68K version yet we have to be able to come up with a 68K * version for the OS at various times. May also need to translate inside * event handlers since the OS will pass the 68K struct to us there. */ #define EVT_DATASIZE_68K 16 /* sez sizeof(event.data) in 68K code */ void evt68k2evtARM( EventType* event, const unsigned char* evt68k ) { event->eType = read_unaligned16( evt68k ); event->penDown = read_unaligned8( evt68k+2 ); event->tapCount = read_unaligned8( evt68k+3 ); event->screenX = read_unaligned16( evt68k+4 ); event->screenY = read_unaligned16( evt68k+6 ); evt68k += 8; /* skip to start of data union */ switch ( event->eType ) { case frmLoadEvent: case frmOpenEvent: case frmCloseEvent: case frmUpdateEvent: XP_ASSERT( &event->data.frmLoad.formID == &event->data.frmClose.formID ); event->data.frmLoad.formID = read_unaligned16( evt68k ); event->data.frmUpdate.updateCode = read_unaligned16( evt68k + 2 ); break; case keyDownEvent: case keyUpEvent: event->data.keyDown.chr = read_unaligned16( evt68k ); event->data.keyDown.keyCode = read_unaligned16( evt68k+2 ); event->data.keyDown.modifiers = read_unaligned16( evt68k+4 ); break; case ctlSelectEvent: event->data.ctlSelect.controlID = read_unaligned16(evt68k); event->data.ctlSelect.pControl = (ControlType*)read_unaligned32(evt68k+2); event->data.ctlSelect.on = (Boolean)read_unaligned8(evt68k+6); event->data.ctlSelect.reserved1 = read_unaligned8(evt68k+7); event->data.ctlSelect.value = read_unaligned16(evt68k+8); break; #ifdef XWFEATURE_FIVEWAY case frmObjectFocusTakeEvent: case frmObjectFocusLostEvent: event->data.frmObjectFocusTake.formID = read_unaligned16(evt68k); event->data.frmObjectFocusTake.objectID = read_unaligned16(evt68k+2); event->data.frmObjectFocusTake.dispatchHint = read_unaligned32(evt68k+4); break; #endif case winExitEvent: case winEnterEvent: XP_ASSERT( &event->data.winEnter.enterWindow == &event->data.winExit.enterWindow ); event->data.winEnter.enterWindow = (WinHandle)read_unaligned32( evt68k ); event->data.winEnter.exitWindow = (WinHandle)read_unaligned32( evt68k+4 ); break; // case penDownEvent: // stuff above is enough // case penMoveEvent: // stuff above is enough // case penUpEvent: // stuff above is enough case menuEvent: event->data.menu.itemID = read_unaligned16( evt68k ); break; case sclRepeatEvent: event->data.sclRepeat.scrollBarID = read_unaligned16( evt68k ); event->data.sclRepeat.pScrollBar = (ScrollBarType*)read_unaligned32( evt68k+2 ); event->data.sclRepeat.value = read_unaligned16( evt68k+6 ); event->data.sclRepeat.newValue = read_unaligned16( evt68k+8 ); event->data.sclRepeat.time = read_unaligned32( evt68k+10 ); break; /* Crosswords events */ case newGameCancelEvent: break; case openSavedGameEvent: ((OpenSavedGameData*)(&event->data.generic))->newGameIndex = read_unaligned16( evt68k ); break; case newGameOkEvent: case loadGameEvent: break; /* case boardRedrawEvt: */ // doResizeWinEvent case prefsChangedEvent: break; default: /* copy the data as binary so we can copy it back later */ memcpy( &event->data, evt68k, EVT_DATASIZE_68K ); break; } } /* evt68k2evtARM */ void evtArm2evt68K( unsigned char* evt68k, const EventType* event ) { write_unaligned16( evt68k, event->eType ); write_unaligned8( evt68k + 2, event->penDown ); write_unaligned8( evt68k + 3, event->tapCount ); write_unaligned16( evt68k + 4, event->screenX ); write_unaligned16( evt68k + 6, event->screenY ); evt68k += 8; switch ( event->eType ) { case frmLoadEvent: case frmOpenEvent: case frmCloseEvent: case frmUpdateEvent: XP_ASSERT( &event->data.frmLoad.formID == &event->data.frmOpen.formID ); XP_ASSERT( &event->data.frmLoad.formID == &event->data.frmClose.formID ); XP_ASSERT( &event->data.frmLoad.formID == &event->data.frmUpdate.formID ); write_unaligned16( evt68k, event->data.frmLoad.formID ); write_unaligned16( evt68k + 2, event->data.frmUpdate.updateCode ); break; case keyDownEvent: case keyUpEvent: write_unaligned16( evt68k, event->data.keyDown.chr ); write_unaligned16( evt68k+2, event->data.keyDown.keyCode ); write_unaligned16( evt68k+4, event->data.keyDown.modifiers ); break; case ctlSelectEvent: write_unaligned16( evt68k, event->data.ctlSelect.controlID ); write_unaligned32( evt68k+2, (unsigned long)event->data.ctlSelect.pControl ); write_unaligned8( evt68k+6, event->data.ctlSelect.on ); write_unaligned8( evt68k+7, event->data.ctlSelect.reserved1 ); write_unaligned16( evt68k+8, event->data.ctlSelect.value ); break; #ifdef XWFEATURE_FIVEWAY case frmObjectFocusTakeEvent: case frmObjectFocusLostEvent: write_unaligned16( evt68k, event->data.frmObjectFocusTake.formID ); write_unaligned16( evt68k+2, event->data.frmObjectFocusTake.objectID ); write_unaligned32( evt68k+4, event->data.frmObjectFocusTake.dispatchHint ); break; #endif case winExitEvent: case winEnterEvent: XP_ASSERT( &event->data.winEnter.enterWindow == &event->data.winExit.enterWindow ); write_unaligned32( evt68k, (unsigned long)event->data.winEnter.enterWindow ); write_unaligned32( evt68k+4, (unsigned long)event->data.winEnter.exitWindow ); break; // case penDownEvent: // stuff above is enough // case penMoveEvent: // stuff above is enough // case penUpEvent: // stuff above is enough case menuEvent: write_unaligned16( evt68k, event->data.menu.itemID ); break; case sclRepeatEvent: write_unaligned16( evt68k, event->data.sclRepeat.scrollBarID ); write_unaligned32( evt68k+2, (unsigned long)event->data.sclRepeat.pScrollBar ); write_unaligned16( evt68k+6, event->data.sclRepeat.value ); write_unaligned16( evt68k+8, event->data.sclRepeat.newValue ); write_unaligned32( evt68k+10, event->data.sclRepeat.time ); break; /* Crosswords events */ case newGameCancelEvent: break; case openSavedGameEvent: write_unaligned16( evt68k, ((OpenSavedGameData*) (&event->data.generic))->newGameIndex ); break; case newGameOkEvent: case loadGameEvent: break; /* case boardRedrawEvt: */ // doResizeWinEvent case prefsChangedEvent: break; default: /* copy the data as binary so we can copy it back later */ memcpy( evt68k, &event->data, EVT_DATASIZE_68K ); break; } } /* evtArm2evt68K */ /* from file SystemMgr.h */ Boolean SysHandleEvent( EventPtr eventP ) { Boolean result; EventType event68K; FUNC_HEADER(SysHandleEvent); evtArm2evt68K( (unsigned char*)&event68K, eventP ); { PNOState* sp = GET_CALLBACK_STATE(); STACK_START(unsigned char, stack, 4); ADD_TO_STACK4(stack, &event68K, 0); STACK_END(stack); result = (Boolean)(*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(sysTrapSysHandleEvent), stack, 4 ); } FUNC_TAIL(SysHandleEvent); return result; } /* SysHandleEvent */ unsigned long handlerEntryPoint( const void* XP_UNUSED(emulStateP), void* userData68KP, Call68KFuncType* XP_UNUSED(call68KFuncP) ) { unsigned long* data = (unsigned long*)userData68KP; FormEventHandlerType* handler = (FormEventHandlerType*)read_unaligned32( (unsigned char*)&data[0] ); PNOState* state = getStorageLoc(); unsigned long oldR10; EventType evtArm; Boolean result; /* set up stack here too? */ asm( "mov %0, r10" : "=r" (oldR10) ); asm( "mov r10, %0" : : "r" (state->gotTable) ); evt68k2evtARM( &evtArm, (unsigned char*)read_unaligned32( (unsigned char*)&data[1]) ); result = (*handler)(&evtArm); asm( "mov r10, %0" : : "r" (oldR10) ); return (unsigned long)result; } /* The stub wants to look like this: static Boolean FormEventHandlerType( EventType *eventP ) { unsigned long data[] = { armEvtHandler, eventP }; return (Boolean)PceNativeCall( handlerEntryPoint, (void*)data ); } */ static unsigned char* makeHandlerStub( FormEventHandlerType* handlerArm ) { unsigned char* stub; unsigned char code_68k[] = { /* 0:*/ 0x4e, 0x56, 0xff, 0xf8, // linkw %fp,#-8 /* 4:*/ 0x20, 0x2e, 0x00, 0x08, // movel %fp@(8),%d0 /* 8:*/ 0x2d, 0x7c, 0x11, 0x22, 0x33, 0x44, // movel #287454020,%fp@(-8) /*14:*/ 0xff, 0xf8, // ????? REQUIRED!!!! /*16:*/ 0x2d, 0x40, 0xff, 0xfc, // movel %d0,%fp@(-4) /*20:*/ 0x48, 0x6e, 0xff, 0xf8, // pea %fp@(-8) /*24:*/ 0x2f, 0x3c, 0x55, 0x66, 0x77, 0x88, // movel #1432778632,%sp@- /*30:*/ 0x4e, 0x4f, // trap #15 /*32:*/ 0xa4, 0x5a, // 0122132 /*34:*/ 0x02, 0x40, 0x00, 0xff, // andiw #255,%d0 /*38:*/ 0x4e, 0x5e, // unlk %fp /*40:*/ 0x4e, 0x75 // rts }; stub = MemPtrNew( sizeof(code_68k) ); memcpy( stub, code_68k, sizeof(code_68k) ); write_unaligned32( &stub[10], /* replace 0x11223344 */ (unsigned long)handlerArm ); write_unaligned32( &stub[26], /* replace 0x55667788 */ (unsigned long)handlerEntryPoint ); /* Need to register this stub so it can be freed (once leaking ceases to be ok on PalmOS) */ return (unsigned char*)stub; } /* makeHandlerStub */ void FrmSetEventHandler( FormType* formP, FormEventHandlerType* handler ) { FUNC_HEADER(FrmSetEventHandler); { PNOState* sp = GET_CALLBACK_STATE(); unsigned char* stub = makeHandlerStub( handler ); STACK_START(unsigned char, stack, 8); ADD_TO_STACK4(stack, formP, 0); ADD_TO_STACK4(stack, stub, 4); STACK_END(stack); (*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(sysTrapFrmSetEventHandler), stack, 8 ); } FUNC_TAIL(FrmSetEventHandler); } /* FrmSetEventHandler */ void flipRect( RectangleType* rout, const RectangleType* rin ) { rout->topLeft.x = Byte_Swap16(rin->topLeft.x); rout->topLeft.y = Byte_Swap16(rin->topLeft.y); rout->extent.x = Byte_Swap16(rin->extent.x); rout->extent.y = Byte_Swap16(rin->extent.y); } /* flipRect */ void flipEngSocketFromArm( unsigned char* sout, const ExgSocketType* sin ) { write_unaligned16( &sout[0], sin->libraryRef ); // UInt16 libraryRef; write_unaligned32( &sout[2], sin->socketRef ); // UInt32 socketRef; write_unaligned32( &sout[6], sin->target ); // UInt32 target; write_unaligned32( &sout[10], sin->count ); // UInt32 count; write_unaligned32( &sout[14], sin->length );// UInt32 length; write_unaligned32( &sout[18], sin->time );// UInt32 time; write_unaligned32( &sout[22], sin->appData ); // UInt32 appData; write_unaligned32( &sout[26], sin->goToCreator ); // UInt32 goToCreator; write_unaligned16( &sout[30], sin->goToParams.dbCardNo ); // UInt16 goToParams.dbCardNo; write_unaligned32( &sout[32], sin->goToParams.dbID ); // LocalID goToParams.dbID; write_unaligned16( &sout[36], sin->goToParams.recordNum ); // UInt16 goToParams.recordNum; write_unaligned32( &sout[38], sin->goToParams.uniqueID ); // UInt32 goToParams.uniqueID; write_unaligned32( &sout[42], sin->goToParams.matchCustom ); // UInt32 goToParams.matchCustom; /* bitfield. All we can do is copy the whole thing, assuming it's 16 bits, and pray that no arm code wants to to use it. */ write_unaligned16( &sout[46], *(UInt16*)((unsigned char*)&sin->goToParams.matchCustom) + sizeof(sin->goToParams.matchCustom) ); write_unaligned32( &sout[48], (unsigned long)sin->description ); // Char *description; write_unaligned32( &sout[52], (unsigned long)sin->type ); // Char *type; write_unaligned32( &sout[56], (unsigned long)sin->name ); // Char *name; } /* flipEngSocketFromArm */ void flipEngSocketToArm( ExgSocketType* sout, const unsigned char* sin ) { sout->libraryRef = read_unaligned16( &sin[0] ); sout->socketRef = read_unaligned32( &sin[2] ); sout->target = read_unaligned32( &sin[6] ); sout->count = read_unaligned32( &sin[10] ); sout->length = read_unaligned32( &sin[14] ); sout->time = read_unaligned32( &sin[18] ); sout->appData = read_unaligned32( &sin[22] ); sout->goToCreator = read_unaligned32( &sin[26] ); sout->goToParams.dbCardNo = read_unaligned16( &sin[30] ); sout->goToParams.dbID = read_unaligned32( &sin[32] ); sout->goToParams.recordNum = read_unaligned16( &sin[36] ); sout->goToParams.uniqueID = read_unaligned32( &sin[38] ); sout->goToParams.matchCustom = read_unaligned32( &sin[42] ); /* bitfield. All we can do is copy the whole thing, assuming it's 16 bits, and pray that no arm code wants to to use it. */ *(UInt16*)(((unsigned char*)&sout->goToParams.matchCustom) + sizeof(sout->goToParams.matchCustom)) = read_unaligned16( &sin[46] ); sout->description = (Char*)read_unaligned32( &sin[48] ); sout->type = (Char*)read_unaligned32( &sin[52] ); sout->name = (Char*)read_unaligned32( &sin[56] ); } /* flipEngSocketToArm */ void flipFileInfoFromArm( unsigned char* fiout, const FileInfoType* fiin ) { write_unaligned32( &fiout[0], fiin->attributes ); write_unaligned32( &fiout[4], (unsigned long)fiin->nameP ); write_unaligned16( &fiout[8], fiin->nameBufLen ); } void flipFileInfoToArm( FileInfoType* fout, const unsigned char* fin ) { fout->attributes = read_unaligned32( &fin[0] ); fout->nameP = (Char*)read_unaligned32( &fin[4] ); fout->nameBufLen = read_unaligned16( &fin[8] ); } /* flipFileInfo */ /* from file List.h */ void LstSetListChoices( ListType* listP, Char** itemsText, Int16 numItems ) { FUNC_HEADER(LstSetListChoices); /* var decls */ /* swapIns */ { XP_U16 i; PNOState* sp = GET_CALLBACK_STATE(); STACK_START(unsigned char, stack, 10); /* pushes */ ADD_TO_STACK4(stack, listP, 0); ADD_TO_STACK4(stack, itemsText, 4); ADD_TO_STACK2(stack, numItems, 8); STACK_END(stack); for ( i = 0; i < numItems; ++i ) { itemsText[i] = (Char*)Byte_Swap32( (unsigned long)itemsText[i] ); } (*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(sysTrapLstSetListChoices), stack, 10 ); /* swapOuts */ } FUNC_TAIL(LstSetListChoices); EMIT_NAME("LstSetListChoices","'L','s','t','S','e','t','L','i','s','t','C','h','o','i','c','e','s'"); } /* LstSetListChoices */ static void params68KtoParamsArm( SysNotifyParamType* paramsArm, const unsigned char* params68K ) { paramsArm->notifyType = read_unaligned32( ¶ms68K[0] ); paramsArm->broadcaster = read_unaligned32( ¶ms68K[4] ); paramsArm->notifyDetailsP = (void*)read_unaligned32( ¶ms68K[8] ); paramsArm->userDataP = (void*)read_unaligned32( ¶ms68K[12] ); paramsArm->handled = read_unaligned8( ¶ms68K[16] ); /* I don't do anything with the data passed in, so no need to swap it... But that'd change for others: make an ARM-corrected copy of the contents of notifyDetailsP if your handler will use it. */ switch( paramsArm->notifyType ) { case sysNotifyVolumeUnmountedEvent: case sysNotifyVolumeMountedEvent: break; #ifdef FEATURE_SILK case sysNotifyDisplayChangeEvent: break; #endif } } /* params68KtoParamsArm */ static void paramsArmtoParams68K( unsigned char* params68K, const SysNotifyParamType* armParams ) { write_unaligned8( ¶ms68K[16], armParams->handled ); } /* paramsArmtoParams68K */ static unsigned long notifyEntryPoint( const void* XP_UNUSED_DBG(emulStateP), void* userData68KP, Call68KFuncType* XP_UNUSED_DBG(call68KFuncP) ) { unsigned long* data = (unsigned long*)userData68KP; SysNotifyProcPtr callback = (SysNotifyProcPtr)read_unaligned32( (unsigned char*)&data[0] ); SysNotifyParamType armParams; PNOState* state = getStorageLoc(); unsigned long oldR10; unsigned char* params68K; Err result; /* set up stack here too? */ asm( "mov %0, r10" : "=r" (oldR10) ); asm( "mov r10, %0" : : "r" (state->gotTable) ); XP_ASSERT( emulStateP == state->emulStateP ); XP_ASSERT( call68KFuncP == state->call68KFuncP ); params68K = (unsigned char*)read_unaligned32((unsigned char*)&data[1]); params68KtoParamsArm( &armParams, params68K ); result = (*callback)(&armParams); /* at least need to write 'handled' back out... */ paramsArmtoParams68K( params68K, &armParams ); asm( "mov r10, %0" : : "r" (oldR10) ); return (unsigned long)result; } /* notifyEntryPoint */ /* The stub wants to look like this: static Err SysNotifyProc(SysNotifyParamType *notifyParamsP) { unsigned long data[] = { armNotifyHandler, notifyParamsP }; return (Err)PceNativeCall( handlerEntryPoint, (void*)data ); } */ static unsigned char* makeNotifyStub( SysNotifyProcPtr callback ) { unsigned char* stub; unsigned char code_68k[] = { /* 0:*/ 0x4e, 0x56, 0xff, 0xf8, // linkw %fp,#-8 /* 4:*/ 0x20, 0x2e, 0x00, 0x08, // movel %fp@(8),%d0 /* 8:*/ 0x2d, 0x7c, 0x11, 0x22, 0x33, 0x44, // movel #287454020,%fp@(-8) /*14:*/ 0xff, 0xf8, // ????? REQUIRED!!!! /*16:*/ 0x2d, 0x40, 0xff, 0xfc, // movel %d0,%fp@(-4) /*20:*/ 0x48, 0x6e, 0xff, 0xf8, // pea %fp@(-8) /*24:*/ 0x2f, 0x3c, 0x55, 0x66, 0x77, 0x88, // movel #1432778632,%sp@- /*30:*/ 0x4e, 0x4f, // trap #15 /*32:*/ 0xa4, 0x5a, // 0122132 /*34:*/ 0x4e, 0x5e, // unlk %fp /*36:*/ 0x4e, 0x75 // rts }; stub = MemPtrNew( sizeof(code_68k) ); memcpy( stub, code_68k, sizeof(code_68k) ); write_unaligned32( &stub[10], /* replace 0x11223344 */ (unsigned long)callback ); write_unaligned32( &stub[26], /* replace 0x55667788 */ (unsigned long)notifyEntryPoint ); /* Need to register this stub so it can be freed (once leaking ceases to be ok on PalmOS) */ return (unsigned char*)stub; } /* makeNotifyStub */ /* from file NotifyMgr.h */ Err SysNotifyRegister( UInt16 cardNo, LocalID dbID, UInt32 notifyType, SysNotifyProcPtr callbackP, Int8 priority, void* userDataP ) { Err result; FUNC_HEADER(SysNotifyRegister); /* var decls */ /* swapIns */ { PNOState* sp = GET_CALLBACK_STATE(); unsigned char* stub = makeNotifyStub( callbackP ); STACK_START(unsigned char, stack, 20); /* pushes */ ADD_TO_STACK2(stack, cardNo, 0); ADD_TO_STACK4(stack, dbID, 2); ADD_TO_STACK4(stack, notifyType, 6); ADD_TO_STACK4(stack, stub, 10); ADD_TO_STACK1(stack, priority, 14); ADD_TO_STACK4(stack, userDataP, 16); STACK_END(stack); result = (Err)(*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(sysTrapSysNotifyRegister), stack, 20 ); /* swapOuts */ } FUNC_TAIL(SysNotifyRegister); EMIT_NAME("SysNotifyRegister","'S','y','s','N','o','t','i','f','y','R','e','g','i','s','t','e','r'"); return result; } /* SysNotifyRegister */ unsigned long listDrawEntryPoint( const void* XP_UNUSED_DBG(emulStateP), void* userData68KP, Call68KFuncType* XP_UNUSED_DBG(call68KFuncP) ) { unsigned long* data = (unsigned long*)userData68KP; ListDrawDataFuncPtr listDrawProc = (ListDrawDataFuncPtr)read_unaligned32( (unsigned char*)&data[0] ); PNOState* state = getStorageLoc(); unsigned long oldR10; Int16 index; char** itemsText; RectangleType rectArm; /* set up stack here too? */ asm( "mov %0, r10" : "=r" (oldR10) ); asm( "mov r10, %0" : : "r" (state->gotTable) ); flipRect( &rectArm, (RectanglePtr)read_unaligned32( (unsigned char*)&data[2] ) ); XP_ASSERT( emulStateP == state->emulStateP ); XP_ASSERT( call68KFuncP == state->call68KFuncP ); index = (Int16)read_unaligned32( (unsigned char*)&data[1] ); itemsText = (char**)read_unaligned32( (unsigned char*)&data[3] ); (*listDrawProc)( index, &rectArm, itemsText ); asm( "mov r10, %0" : : "r" (oldR10) ); return 0L; /* no result to return */ } /* listDrawEntryPoint */ static unsigned char* makeListDrawStub( ListDrawDataFuncPtr func ) { /* called function looks like this: void listDrawFunc(Int16 index, RectanglePtr bounds, char** itemsText) { unsigned long data[] = { func, index, bounds, itemsText }; return (Err)PceNativeCall( listDrawEntryPoint, (void*)data ); } */ unsigned char* stub; unsigned char code_68k[] = { /* 0:*/ 0x4e, 0x56, 0xff, 0xf0, // linkw %fp,#-16 /* 4:*/ 0x30, 0x2e, 0x00, 0x08, // movew %fp@(8),%d0 /* 8:*/ 0x22, 0x2e, 0x00, 0x0a, // movel %fp@(10),%d1 /* c:*/ 0x24, 0x2e, 0x00, 0x0e, // movel %fp@(14),%d2 /*10:*/ 0x2d, 0x7c, 0x11, 0x22,0x33,0x44,// movel #287454020,%fp@(-16) /*16:*/ 0xff, 0xf0, /*18:*/ 0x30, 0x40, // moveaw %d0,%a0 /*1a:*/ 0x2d, 0x48, 0xff, 0xf4, // movel %a0,%fp@(-12) /*1e:*/ 0x2d, 0x41, 0xff, 0xf8, // movel %d1,%fp@(-8) /*22:*/ 0x2d, 0x42, 0xff, 0xfc, // movel %d2,%fp@(-4) /*26:*/ 0x48, 0x6e, 0xff, 0xf0, // pea %fp@(-16) /*2a:*/ 0x2f, 0x3c, 0x55, 0x66, 0x77, 0x88, // movel #1432778632,%sp@- /*30:*/ 0x4e, 0x4f, // trap #15 /*32:*/ 0xa4, 0x5a, // 0122132 /*34:*/ 0x4e, 0x5e, // unlk %fp /*36:*/ 0x4e, 0x75 // rts }; stub = MemPtrNew( sizeof(code_68k) ); memcpy( stub, code_68k, sizeof(code_68k) ); write_unaligned32( &stub[0x12], /* replace 0x11223344 */ (unsigned long)func ); write_unaligned32( &stub[0x2c], /* replace 0x55667788 */ (unsigned long)listDrawEntryPoint ); return (unsigned char*)stub; } /* makeListDrawStub */ /* from file List.h */ void LstSetDrawFunction( ListType* listP, ListDrawDataFuncPtr func ) { FUNC_HEADER(LstSetDrawFunction); /* var decls */ /* swapIns */ { PNOState* sp = GET_CALLBACK_STATE(); unsigned char* stub = makeListDrawStub( func ); STACK_START(unsigned char, stack, 8); /* pushes */ ADD_TO_STACK4(stack, listP, 0); ADD_TO_STACK4(stack, stub, 4); STACK_END(stack); (*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(sysTrapLstSetDrawFunction), stack, 8 ); /* swapOuts */ } FUNC_TAIL(LstSetDrawFunction); EMIT_NAME("LstSetDrawFunction","'L','s','t','S','e','t','D','r','a','w','F','u','n','c','t','i','o','n'"); } /* LstSetDrawFunction */ void flipDateTimeToArm( DateTimeType* out, const unsigned char* in ) { const DateTimeType* inp = (DateTimeType*)in; out->second = Byte_Swap16( inp->second ); out->minute = Byte_Swap16( inp->minute ); out->hour = Byte_Swap16( inp->hour ); out->day = Byte_Swap16( inp->day ); out->month = Byte_Swap16( inp->month ); out->year = Byte_Swap16( inp->year ); out->weekDay = Byte_Swap16( inp->weekDay ); } /* from file NetMgr.h */ NetHostInfoPtr NetLibGetHostByName( UInt16 libRefNum, const Char* nameP, NetHostInfoBufPtr bufP, Int32 timeout, Err* errP ) { NetHostInfoPtr result; FUNC_HEADER(NetLibGetHostByName); /* var decls */ /* swapIns */ { PNOState* sp = GET_CALLBACK_STATE(); STACK_START(unsigned char, stack, 18); /* pushes */ ADD_TO_STACK2(stack, libRefNum, 0); ADD_TO_STACK4(stack, nameP, 2); ADD_TO_STACK4(stack, bufP, 6); ADD_TO_STACK4(stack, timeout, 10); ADD_TO_STACK4(stack, errP, 14); STACK_END(stack); result = (NetHostInfoPtr) (*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(netLibTrapGetHostByName), stack, kPceNativeWantA0 | 18 ); /* swapOuts */ if ( result != NULL ) { result->nameP = Byte_Swap32( result->nameP ); result->nameAliasesP = Byte_Swap32( result->nameAliasesP ); result->addrType = Byte_Swap16( result->addrType ); result->addrLen = Byte_Swap16( result->addrLen ); result->addrListP = Byte_Swap32( result->addrListP ); /* we'll use only the first */ result->addrListP[0] = Byte_Swap32( result->addrListP[0] ); *(XP_U32*)result->addrListP[0] = Byte_Swap32( *(XP_U32*)result->addrListP[0] ); } } FUNC_TAIL(NetLibGetHostByName); EMIT_NAME("NetLibGetHostByName","'N','e','t','L','i','b','G','e','t','H','o','s','t','B','y','N','a','m','e'"); return result; } /* NetLibGetHostByName */ static unsigned long exgWriteEntry( const void* XP_UNUSED_DBG(emulStateP), void* userData68KP, Call68KFuncType* XP_UNUSED_DBG(call68KFuncP) ) { unsigned long* data = (unsigned long*)userData68KP; unsigned long oldR10; ExgDBWriteProcPtr exgProc = (ExgDBWriteProcPtr)read_unaligned32( (unsigned char*)&data[0] ); PNOState* state = getStorageLoc(); UInt32* sizeP; Err err; /* set up stack here too? */ asm( "mov %0, r10" : "=r" (oldR10) ); asm( "mov r10, %0" : : "r" (state->gotTable) ); XP_ASSERT( emulStateP == state->emulStateP ); XP_ASSERT( call68KFuncP == state->call68KFuncP ); sizeP = (UInt32*)read_unaligned32( (unsigned char*)&data[2] ); SWAP4_NON_NULL_OUT(sizeP); err = (*exgProc)( (const void*)read_unaligned32((unsigned char*)&data[1]), sizeP, (void*)read_unaligned32( (unsigned char*)&data[3] ) ); SWAP4_NON_NULL_IN(sizeP); asm( "mov r10, %0" : : "r" (oldR10) ); return (unsigned long)err; /* no result to return */ } /* exgWriteEntry */ static void makeExgWriteStub( ExgDBWriteProcPtr proc, unsigned char* stub, XP_U16 XP_UNUSED_DBG(stubSize) ) { /* Err ExgDBWriteProc( const void *dataP, UInt32 *sizeP, void *userDataP) */ /* { */ /* unsigned long data[] = { */ /* (unsigned long)0x11223344, */ /* (unsigned long)dataP, */ /* (unsigned long)sizeP, */ /* (unsigned long)userDataP */ /* }; */ /* return (Err)PceNativeCall( (void*)0x55667788, (void*)data ); */ /* } */ unsigned char code_68k[] = { /* 0:*/ 0x4e, 0x56, 0xff, 0xf0, /* linkw %fp,#-16 */ /* 4:*/ 0x20, 0x2e, 0x00, 0x08, /* movel %fp@(8),%d0 */ /* 8:*/ 0x22, 0x2e, 0x00, 0x0c, /* movel %fp@(12),%d1 */ /* c:*/ 0x24, 0x2e, 0x00, 0x10, /* movel %fp@(16),%d2 */ /*10:*/ 0x2d, 0x7c, 0x11, 0x22, 0x33, 0x44,/*movel #287454020,%fp@(-16)*/ /*16:*/ 0xff, 0xf0, /*18:*/ 0x2d, 0x40, 0xff, 0xf4, /* movel %d0,%fp@(-12) */ /*1c:*/ 0x2d, 0x41, 0xff, 0xf8, /* movel %d1,%fp@(-8) */ /*20:*/ 0x2d, 0x42, 0xff, 0xfc, /* movel %d2,%fp@(-4) */ /*24:*/ 0x48, 0x6e, 0xff, 0xf0, /* pea %fp@(-16) */ /*28:*/ 0x2f, 0x3c, 0x55, 0x66, 0x77, 0x88, /* movel #1432778632,%sp@- */ /*2e:*/ 0x4e, 0x4f, /* trap #15 */ /*30:*/ 0xa4, 0x5a, /* 0122132 */ /*32:*/ 0x4e, 0x5e, /* unlk %fp */ /*34:*/ 0x4e, 0x75 /* rts */ }; XP_ASSERT( sizeof(code_68k) <= stubSize ); memcpy( stub, code_68k, sizeof(code_68k) ); write_unaligned32( &stub[0x12], /* replace 0x11223344 */ (unsigned long)proc ); write_unaligned32( &stub[0x2a], /* replace 0x55667788 */ (unsigned long)exgWriteEntry ); } /* makeExgWriteStub */ /* from file ExgMgr.h */ Err ExgDBWrite( ExgDBWriteProcPtr writeProcP, void* userDataP, const char* nameP, LocalID dbID, UInt16 cardNo ) { Err result; FUNC_HEADER(ExgDBWrite); /* var decls */ /* swapIns */ { PNOState* sp = GET_CALLBACK_STATE(); unsigned char stub[0x36]; STACK_START(unsigned char, stack, 18); makeExgWriteStub( writeProcP, stub, sizeof(stub) ); /* pushes */ ADD_TO_STACK4(stack, stub, 0); ADD_TO_STACK4(stack, userDataP, 4); ADD_TO_STACK4(stack, nameP, 8); ADD_TO_STACK4(stack, dbID, 12); ADD_TO_STACK2(stack, cardNo, 16); STACK_END(stack); result = (Err)(*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(sysTrapExgDBWrite), stack, 18 ); /* swapOuts */ } FUNC_TAIL(ExgDBWrite); EMIT_NAME("ExgDBWrite","'E','x','g','D','B','W','r','i','t','e'"); return result; } /* ExgDBWrite */ #ifdef XWFEATURE_BLUETOOTH static void btLibManagementEventType68K_TO_ARM( BtLibManagementEventType* out, const unsigned char* in ) { BtLibManagementEventEnum event = (BtLibManagementEventEnum)read_unaligned8( &in[0] ); out->event = event; out->status = read_unaligned16( &in[2] ); switch( event ) { case btLibManagementEventACLConnectInbound: case btLibManagementEventACLDisconnect: memcpy( &out->eventData.bdAddr, &in[4], sizeof(out->eventData.bdAddr) ); break; case btLibManagementEventAccessibilityChange: out->eventData.accessible = (BtLibAccessibleModeEnum)read_unaligned8( &in[4] ); break; } } static unsigned long btLibManagementProcArmEntry( const void* XP_UNUSED_DBG(emulStateP), void* userData68KP, Call68KFuncType* XP_UNUSED_DBG(call68KFuncP) ) { unsigned long* data; BtLibManagementProcPtr procPtr; PNOState* state; unsigned long oldR10; BtLibManagementEventType mEvent; const BtLibManagementEventType* mEventP; UInt32 refCon; data = (unsigned long*)userData68KP; state = getStorageLoc(); /* set up stack here too? */ asm( "mov %0, r10" : "=r" (oldR10) ); asm( "mov r10, %0" : : "r" (state->gotTable) ); XP_ASSERT( emulStateP == state->emulStateP ); XP_ASSERT( call68KFuncP == state->call68KFuncP ); procPtr = (BtLibManagementProcPtr)read_unaligned32( &data[0] ); mEventP = read_unaligned32( &data[1] ); btLibManagementEventType68K_TO_ARM( &mEvent, (unsigned char*)mEventP ); refCon = read_unaligned32( (unsigned char*)&data[2] ); (*procPtr)( &mEvent, refCon ); asm( "mov r10, %0" : : "r" (oldR10) ); return 0L; /* no result to return */ } /* btLibManagementProcArmEntry */ /* BtLibSocketProcPtr and BtLibManagementProc have the same signatures as far as 68K code goes: */ /* static void BtLibManagementProc( BtLibManagementEventType *mEvent, UInt32 refCon ) */ /* { */ /* unsigned long data[] = { callbackP, mEvent, refCon }; */ /* (void)PceNativeCall( btLibManagementProcArmEntry, (void*)data ); */ /* } */ static unsigned char* make44stub( void* callbackP, NativeFuncType armFunc ) { unsigned char* stub; unsigned char code_68k[] = { /* 0 */ 0x4e, 0x56, 0xff, 0xf4, // linkw %fp,#-12 /* 4 */ 0x20, 0x2e, 0x00, 0x08, // movel %fp@(8),%d0 /* 8 */ 0x22, 0x2e, 0x00, 0x0c, // movel %fp@(12),%d1 /* C */ 0x2d, 0x7c, 0x11, 0x22, 0x33, 0x44, // movel #287454020,%fp@(-12) /* 12 */ 0xff, 0xf4, /* 14 */ 0x2d, 0x40, 0xff, 0xf8, // movel %d0,%fp@(-8) /* 18 */ 0x2d, 0x41, 0xff, 0xfc, // movel %d1,%fp@(-4) /* 1C */ 0x48, 0x6e, 0xff, 0xf4, // pea %fp@(-12) /* 20 */ 0x2f, 0x3c, 0x55, 0x66, 0x77, 0x88, // movel #1432778632,%sp@- /* 26 */ 0x4e, 0x4f, // trap #15 /* 28 */ 0xa4, 0x5a, // 0122132 /* 2A */ 0x50, 0x8f, // addql #8,%sp /* 2C */ 0x4e, 0x5e, // unlk %fp /* 2E */ 0x4e, 0x75 // rts }; stub = MemPtrNew( sizeof(code_68k) ); memcpy( stub, code_68k, sizeof(code_68k) ); write_unaligned32( &stub[0x0E], /* replace 0x11223344 */ (unsigned long)callbackP ); write_unaligned32( &stub[0x22], /* replace 0x55667788 */ (unsigned long)armFunc ); return stub; } /* findOrMakeBTProcStub */ static void btLibSocketEventType68K_TO_ARM( BtLibSocketEventType* out, const unsigned char* in ) { BtLibSocketEventEnum event = read_unaligned8( &in[0] ); /* enum? */ out->event = event; out->socket = read_unaligned16( &in[2] ); out->status = read_unaligned16( &in[4] ); switch( event ) { case btLibSocketEventConnectedInbound: out->eventData.newSocket = read_unaligned16( &in[6] ); break; case btLibSocketEventData: out->eventData.data.dataLen = read_unaligned16( &in[6] ); out->eventData.data.data = read_unaligned32( &in[8] ); break; default: /* shut up, compiler */ break; } } static unsigned long btSocketProcArmEntry( const void* XP_UNUSED_DBG(emulStateP), void* userData68KP, Call68KFuncType* XP_UNUSED_DBG(call68KFuncP) ) { BtLibSocketEventType sEvent; BtLibSocketEventType* sEventP; UInt32 refCon; unsigned long* data = (unsigned long*)userData68KP; unsigned long oldR10; BtLibSocketProcPtr procPtr = (BtLibSocketProcPtr)read_unaligned32( (unsigned char*)&data[0] ); PNOState* state = getStorageLoc(); /* set up stack here too? */ asm( "mov %0, r10" : "=r" (oldR10) ); asm( "mov r10, %0" : : "r" (state->gotTable) ); XP_ASSERT( emulStateP == state->emulStateP ); XP_ASSERT( call68KFuncP == state->call68KFuncP ); sEventP = (BtLibSocketEventType*) read_unaligned32( (unsigned char*)&data[1] ); btLibSocketEventType68K_TO_ARM( &sEvent, (unsigned char*)sEventP ); refCon = read_unaligned32( (unsigned char*)&data[2] ); (*procPtr)( &sEvent, refCon ); asm( "mov r10, %0" : : "r" (oldR10) ); return 0L; /* no result to return */ } /* btSocketProcArmEntry */ /* from file BtLib.h */ Err BtLibRegisterManagementNotification( UInt16 btLibRefNum, BtLibManagementProcPtr callbackP, UInt32 refCon ) { Err result; FUNC_HEADER(BtLibRegisterManagementNotification); /* var decls */ /* swapIns */ { PNOState* sp = GET_CALLBACK_STATE(); unsigned char* stub = make44stub( callbackP, btLibManagementProcArmEntry ); result = FtrSet( APPID, PACE_BT_CBK_FEATURE, stub ); XP_LOGF( "%s: stub=%lx", __func__, stub ); STACK_START(unsigned char, stack, 10); /* pushes */ ADD_TO_STACK2(stack, btLibRefNum, 0); ADD_TO_STACK4(stack, stub, 2); ADD_TO_STACK4(stack, refCon, 6); STACK_END(stack); result = (Err)(*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(btLibTrapRegisterManagementNotification), stack, 10 ); /* swapOuts */ } FUNC_TAIL(BtLibRegisterManagementNotification); EMIT_NAME("BtLibRegisterManagementNotification","'B','t','L','i','b','R','e','g','i','s','t','e','r','M','a','n','a','g','e','m','e','n','t','N','o','t','i','f','i','c','a','t','i','o','n'"); return result; } /* BtLibRegisterManagementNotification */ /* from file BtLib.h */ Err BtLibUnregisterManagementNotification( UInt16 btLibRefNum, BtLibManagementProcPtr XP_UNUSED_DBG(callbackP) ) { Err result; unsigned char* stub = NULL; FUNC_HEADER(BtLibUnregisterManagementNotification); result = FtrGet( APPID, PACE_BT_CBK_FEATURE, (UInt32*)&stub ); XP_ASSERT( !!stub && errNone == result ); XP_LOGF( "%s: stub=%lx", __func__, stub ); #ifdef DEBUG { XP_U32 bkwrds; write_unaligned32( &bkwrds, callbackP ); XP_ASSERT( 0 == XP_MEMCMP( &bkwrds, &stub[0x0E], sizeof(callbackP) ) ); } #endif /* var decls */ /* swapIns */ { PNOState* sp = GET_CALLBACK_STATE(); STACK_START(unsigned char, stack, 6); /* pushes */ ADD_TO_STACK2(stack, btLibRefNum, 0); ADD_TO_STACK4(stack, stub, 2); STACK_END(stack); result = (Err)(*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(btLibTrapUnregisterManagementNotification), stack, 6 ); /* swapOuts */ } MemPtrFree( stub ); XP_LOGF( "%s: stub freed", __func__ ); FUNC_TAIL(BtLibUnregisterManagementNotification); EMIT_NAME("BtLibUnregisterManagementNotification","'B','t','L','i','b','U','n','r','e','g','i','s','t','e','r','M','a','n','a','g','e','m','e','n','t','N','o','t','i','f','i','c','a','t','i','o','n'"); return result; } /* BtLibUnregisterManagementNotification */ /* from file BtLib.h */ Err BtLibSocketCreate( UInt16 btLibRefNum, BtLibSocketRef* socketRefP, BtLibSocketProcPtr callbackP, UInt32 refCon, BtLibProtocolEnum socketProtocol ) { Err result; FUNC_HEADER(BtLibSocketCreate); /* var decls */ /* swapIns */ /* SWAP2_NON_NULL_IN(socketRefP); */ { PNOState* sp = GET_CALLBACK_STATE(); unsigned char* stub = make44stub( callbackP, btSocketProcArmEntry ); STACK_START(unsigned char, stack, 16); /* pushes */ ADD_TO_STACK2(stack, btLibRefNum, 0); ADD_TO_STACK4(stack, socketRefP, 2); ADD_TO_STACK4(stack, stub, 6); ADD_TO_STACK4(stack, refCon, 10); ADD_TO_STACK1(stack, socketProtocol, 14); STACK_END(stack); result = (Err) (*sp->call68KFuncP)( sp->emulStateP, PceNativeTrapNo(btLibTrapSocketCreate), stack, 16 ); /* swapOuts */ SWAP2_NON_NULL_OUT(socketRefP); } FUNC_TAIL(BtLibSocketCreate); EMIT_NAME("BtLibSocketCreate","'B','t','L','i','b','S','o','c','k','e','t','C','r','e','a','t','e'"); return result; } /* BtLibSocketCreate */ void flipBtConnInfoArm268K( unsigned char* out, const BtLibSocketConnectInfoType* in ) { write_unaligned32( out + 0, in->remoteDeviceP ); write_unaligned16( out + 4, in->data.L2Cap.remotePsm ); write_unaligned16( out + 6, in->data.L2Cap.minRemoteMtu ); write_unaligned16( out + 8, in->data.L2Cap.localMtu ); } /* from file BtLib.h */ void flipBtSocketListenInfoArm268K( unsigned char* out, const BtLibSocketListenInfoType* in) { /* Some moron didn't put a type field in BtLibSocketListenInfoType. The interpretation of the data depends on the protocol with which the socket (param to BtLibSocketListen) was created. We *could* pass the socket into this function and pull the protocol out, using that to decide which overlapping union field to swap, but let's just hard-code btLibL2CapProtocol (field data.L2Cap) since that's what I'm using.*/ write_unaligned16( out + 0, in->data.L2Cap.localPsm ); write_unaligned16( out + 2, in->data.L2Cap.localMtu ); write_unaligned16( out + 4, in->data.L2Cap.minRemoteMtu ); } #endif /* XWFEATURE_BLUETOOTH */