/* Legal: Version: MPL 1.1 The contents of this file are subject to the Mozilla Public License Version 1.1 the "License"; you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.mozilla.org/MPL/ Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the specific language governing rights and limitations under the License. The Original Code is the YSI framework. The Initial Developer of the Original Code is Alex "Y_Less" Cole. Portions created by the Initial Developer are Copyright C 2011 the Initial Developer. All Rights Reserved. Contributors: Y_Less koolk JoeBullet/Google63 g_aSlice/Slice Misiur samphunter tianmeta maddinat0r spacemud Crayder Dayvison Ahmad45123 Zeex irinel1996 Yiin- Chaprnks Konstantinos Masterchen09 Southclaws PatchwerkQWER m0k1 paulommu udan111 Thanks: JoeBullet/Google63 - Handy arbitrary ASM jump code using SCTRL. ZeeX - Very productive conversations. koolk - IsPlayerinAreaEx code. TheAlpha - Danish translation. breadfish - German translation. Fireburn - Dutch translation. yom - French translation. 50p - Polish translation. Zamaroht - Spanish translation. Los - Portuguese translation. Dracoblue, sintax, mabako, Xtreme, other coders - Producing other modes for me to strive to better. Pixels^ - Running XScripters where the idea was born. Matite - Pestering me to release it and using it. Very special thanks to: Thiadmer - PAWN, whose limits continue to amaze me! Kye/Kalcor - SA:MP. SA:MP Team past, present and future - SA:MP. Optional plugins: Gamer_Z - GPS. Incognito - Streamer. Me - sscanf2, fixes2, Whirlpool. */ /* ad88888ba d8" "8b ,d Y8, 88 `Y8aaaaa, ,adPPYba, MM88MMM 88 88 8b,dPPYba, `"""""8b, a8P_____88 88 88 88 88P' "8a `8b 8PP""""""" 88 88 88 88 d8 Y8a a8P "8b, ,aa 88, "8a, ,a88 88b, ,a8" "Y88888P" `"Ybbd8"' "Y888 `"YbbdP'Y8 88`YbbdP"' 88 88 */ enum e_INI_LINE_TYPE { e_INI_LINE_TYPE_INVALID, e_INI_LINE_TYPE_DATALESS, e_INI_LINE_TYPE_ENTRY, e_INI_LINE_TYPE_CONT, e_INI_LINE_TYPE_TAG } static stock YSI_g_sCurLine[MAX_INI_LINE]; /* 88b d88 888b d888 88`8b d8'88 88 `8b d8' 88 ,adPPYYba, ,adPPYba, 8b,dPPYba, ,adPPYba, ,adPPYba, 88 `8b d8' 88 "" `Y8 a8" "" 88P' "Y8 a8" "8a I8[ "" 88 `8b d8' 88 ,adPPPPP88 8b 88 8b d8 `"Y8ba, 88 `888' 88 88, ,88 "8a, ,aa 88 "8a, ,a8" aa ]8I 88 `8' 88 `"8bbdP"Y8 `"Ybbd8"' 88 `"YbbdP"' `"YbbdP"' */ #define INI_Parse(%1,%2) \ forward @INI_%1_%2(name[], value[]); \ @INI_%1_%2(name[], value[]) #define INI:%0[%1](%2) \ forward @INI_%0_%1(%2); \ @INI_%0_%1(%2) /*-------------------------------------------------------------------------*//** * Name of the INI key. * Variable to fill with integer value. *//*------------------------------------------------------------------------**/ P:D(INI_Int(const name[],&variable)); #define INI_Int(%1,%2) \ if (!strcmp((%1), name, true)) return %2 = strval(value) /*-------------------------------------------------------------------------*//** * Name of the INI key. * Variable to fill with float value. *//*------------------------------------------------------------------------**/ P:D(INI_Float(const name[],&Float:variable)); #define INI_Float(%1,%2) \ if (!strcmp((%1), name, true)) return _:(%2 = floatstr(value)) /*-------------------------------------------------------------------------*//** * Name of the INI key. * Variable to fill with hex value. *//*------------------------------------------------------------------------**/ P:D(INI_Hex(const name[],&variable)); #define INI_Hex(%1,%2) \ if (!strcmp((%1), name, true)) return %2 = hexstr(value) /*-------------------------------------------------------------------------*//** * Name of the INI key. * Variable to fill with binary value. *//*------------------------------------------------------------------------**/ P:D(INI_Bin(const name[],&variable)); #define INI_Bin(%1,%2) \ if (!strcmp((%1), name, true)) return %2 = binstr(value) /*-------------------------------------------------------------------------*//** * Name of the INI key. * Variable to fill with string value. * Optional string length. * * The old version of "INI_String" didn't like not having a length. It gave a * very odd error message too. This has now been corrected by making the * length parameter optional. * *//*------------------------------------------------------------------------**/ P:D(INI_String(const name[],variable[])); #define INI_String(%1,%2) \ if (!strcmp((%1), name, true)) return _:INI_CHECK_LEN:strcpy(%2, value) #define INI_CHECK_LEN:strcpy(%0,%1,%2) strcpy(%0,%2,%1) /*-------------------------------------------------------------------------*//** * Name of the INI key. * Variable to fill with string value. *//*------------------------------------------------------------------------**/ P:D(INI_Bool(const name[],&bool:variable)); #define INI_Bool(%1,%2) \ if (!strcmp((%1), name, true)) return %2 = boolstr(value) /* 88 88 88 88 88 88 ,d "" 88 88 88 88 88 88 88 MM88MMM 88 88 ,adPPYba, 88 88 88 88 88 I8[ "" 88 88 88 88 88 `"Y8ba, Y8a. .a8P 88, 88 88 aa ]8I `"Y8888Y"' "Y888 88 88 `"YbbdP"' */ #define INI_SkipWhitespace(%0,%1) while ('\0' < %0[%1] <= ' ') ++%1 /*-------------------------------------------------------------------------*//** * The callback destination. * The function name format. * The tag destination. * The tag source. * The callback parameter specifiers. * Use "CallRemoteFunction". * * Was the function found? * * * Gets a callback given a partial function name and a tag name. Also saves * the tag elsewhere. This might not work as a separate function - it will * need to be in the function called by the function with the inlines in. * *//*------------------------------------------------------------------------**/ P:D(INI_GetCallback(callback,format,tag,input,callbackFormat,remote)); #define INI_GetCallback(%0,%1,%2,%3,%4,%5) \ ( \ strcpy((%2), (%3)), \ Inline_Reset((%0)), \ format(YSI_g_sCurLine, sizeof (YSI_g_sCurLine), (%1), (%2)), \ Callback_Get(callback_tag:YSI_g_sCurLine, (%0), (%4), (%5)) \ ) /* 88b d88 88 db 88888888ba 88 888b d888 "" d88b 88 "8b 88 88`8b d8'88 d8'`8b 88 ,8P 88 88 `8b d8' 88 ,adPPYYba, 88 8b,dPPYba, d8' `8b 88aaaaaa8P' 88 88 `8b d8' 88 "" `Y8 88 88P' `"8a d8YaaaaY8b 88""""""' 88 88 `8b d8' 88 ,adPPPPP88 88 88 88 d8""""""""8b 88 88 88 `888' 88 88, ,88 88 88 88 d8' `8b 88 88 88 `8' 88 `"8bbdP"Y8 88 88 88 d8' `8b 88 88 */ /*-------------------------------------------------------------------------*//** * The file to load. * The format string to generate the remote function to * pass the data to once loaded. * The order of the remoteFormat parameters. * Send additional data. * Additional data to send. * Call local functions instead of global ones. * Pass the tag as an extra parameter not the function * name. * Apply the tag name filter to all tags or just prefixed * ones? * Text to use to search for which tags to load. * * bFileFirst sets the order and inclusion of the possible remoteFormat * parameters. If true the format will add the filename first then the * current tag, if false the order will be reversed. This can also be used * to exclude one or the other from the function name by setting the required * parameter to be entered first and then only having one %s in the format * sting. The default order is tag first for languages compatibility. * * This function is now EXTENSIVELY documented here: * * * * *//*------------------------------------------------------------------------**/ stock bool:INI_ParseFile(fname[], remoteFormat[], bool:bFileFirst = false, bool:bExtra = false, extra = 0, bool:bLocal = true, bool:bPassTag = false, bool:bFilter = true, filter[] = "") { P:3("bool:INI_ParseFile called: \"%s\", \"%s\", %i, %i, %i, %i, %i, %i, \"%s\"", fname, remoteFormat, bFileFirst, bExtra, extra, bLocal, bPassTag, bFilter, filter); static callback[E_CALLBACK_DATA], tag[32], // The current tag being parsed. cbSpec[5], // The callback specifier. cbName[32]; // The callback name format. new File:f = fopen(fname, io_read); if (!f) return false; new rlen, ppos, p0s, p0e, p1s, p1e, p2s, p2e; P:5("INI_ParseFile: Opened."); bLocal = !bLocal, // Invert for "remote". INI_MakeCallbackFormat(bExtra, bPassTag, cbSpec), INI_SetupCallbackName(cbName, remoteFormat, fname, bFileFirst); new bool:handle = INI_GetCallback(callback, cbName, tag, "", cbSpec, bLocal); while ((rlen = fread(f, YSI_g_sCurLine))) { ppos += rlen; switch (INI_IdentifyLineType(YSI_g_sCurLine, p0s, p0e, p1s, p1e, p2s, p2e)) { case e_INI_LINE_TYPE_INVALID: { // "P:I" not "P:W" because this is not a script issue. P:I("Invalid line in INI file \"%s\": %s", fname, YSI_g_sCurLine); } case e_INI_LINE_TYPE_DATALESS: { // Do nothing. } case e_INI_LINE_TYPE_TAG: { // A new tag. Callback_Restore(callback), Callback_Release(callback), // First, check if it is a tag we might care about. YSI_g_sCurLine[p0e] = '\0'; if (YSI_g_sCurLine[p0s] == '@' && YSI_g_sCurLine[p0s + 1] == '@' && (p0e = strfind(YSI_g_sCurLine, "-")) != -1) { // Check if the current tag is one of the ones we want to // filer for. The "@@" prefix is the "filterable" prefix. // If there is no filter then everything will be loaded. if (p0e == p0s + 2 || !YSI_g_sCurLine[p0e + 1]) { P:I("Invalid line in INI file \"%s\": %s", fname, YSI_g_sCurLine); continue; } YSI_g_sCurLine[p0e] = '\0'; if (bFilter && strcmp(YSI_g_sCurLine[p0s + 2], filter)) { // Only skip this if filtering is enabled. We can't put // the "bFilter" check higher or the "-" won't be found // to resolve the tag name accurately. handle = false; continue; } p0s = p0e + 1; } // This is a tag we can use, is there a handler for it? // Is this based on another tag? If so recurse and do that one // first. new parent[32]; YSI_g_sCurLine[p1e] = '\0', strcat(parent, YSI_g_sCurLine[p1s]); if ((handle = INI_GetCallback(callback, cbName, tag, YSI_g_sCurLine[p0s], cbSpec, bLocal)) && p1s != p1e) { // No point recursing if there's no handler is there? if (!INI_DoParentTag(ppos, parent, f, callback, bExtra, extra, bPassTag, tag, bFilter, filter)) P:I("Invalid hierarchy in INI file: \"%s\" for tag \"%s\"", fname, tag); fseek(f, ppos, seek_start); } } default: // e_INI_LINE_TYPE_ENTRY and e_INI_LINE_TYPE_CONT. { if (!handle) continue; // Standard key-value pair. YSI_g_sCurLine[p0e] = YSI_g_sCurLine[p1e] = '\0'; if (bExtra) { if (bPassTag) Callback_Call(callback, extra, tag, YSI_g_sCurLine[p0s], YSI_g_sCurLine[p1s]); else Callback_Call(callback, extra, YSI_g_sCurLine[p0s], YSI_g_sCurLine[p1s]); } else { if (bPassTag) Callback_Call(callback, tag, YSI_g_sCurLine[p0s], YSI_g_sCurLine[p1s]); else Callback_Call(callback, YSI_g_sCurLine[p0s], YSI_g_sCurLine[p1s]); } } } // Don't put any code down here (at the end of the loop). } return Callback_Restore(callback), Callback_Release(callback), fclose(f), true; } /*-------------------------------------------------------------------------*//** * The file to load. * Send additional data. * Additional data to send. * Call local functions instead of gloabal ones. * * INI_ParseFile * * * Wrapper for INI_ParseFile to use standard API features so people can * worry even less. Designed for use with INI_Parse. * *//*------------------------------------------------------------------------**/ stock bool:INI_Load(filename[], bool:bExtra = false, extra = 0, bool:bLocal = true) { P:3("bool:INI_Load called: \"%s\", %i, %i, %i", filename, _:bExtra, extra, _:bLocal); return INI_ParseFile(filename, "@INI_%s_%s", .bFileFirst = true, .bExtra = bExtra, .extra = extra, .bLocal = bLocal); } /* 88 88 88 ad88 88 88 ,d "" d8" 88 88 88 88 88 ,adPPYb,88 ,adPPYba, 8b,dPPYba, MM88MMM 88 MM88MMM 8b d8 88 a8" `Y88 a8P_____88 88P' `"8a 88 88 88 `8b d8' 88 8b 88 8PP""""""" 88 88 88 88 88 `8b d8' 88 "8a, ,d88 "8b, ,aa 88 88 88, 88 88 `8b,d8' 88 `"8bbdP"Y8 `"Ybbd8"' 88 88 "Y888 88 88 Y88' d8' d8' */ /*-------------------------------------------------------------------------*//** * The string you want to type analyse. * Start of part 0. * End of part 0. * Start of part 1. * End of part 1. * Start of part 2. * End of part 2. * Is this a line continuation? * * e_INI_LINE_TYPE * * * This function's signature is so long that I put it on a separate line. This * takes a line and determines what it is and where the parts are. * *//*------------------------------------------------------------------------**/ stock e_INI_LINE_TYPE: INI_IdentifyLineType(const str[], &p0s, &p0e, &p1s, &p1e, &p2s, &p2e) { // Reset everything. p0s = p0e = p1s = p1e = p2s = p2e = 0; // Do this purely with a single main loop, and a state machine. new end, e_INI_LINE_TYPE:ret = e_INI_LINE_TYPE_DATALESS, pos; INI_SkipWhitespace(str, pos); switch (str[pos++]) { case '\0': return e_INI_LINE_TYPE_DATALESS; case ';': goto state_in_comment; case '[': goto state_in_tag; case '\\': if (str[pos]) { p0s = pos - 1, end = ++pos; } else return e_INI_LINE_TYPE_INVALID; case '=': return e_INI_LINE_TYPE_INVALID; default: p0s = pos - 1; } // Default end point, for single-character lines. end = pos; //state_in_entry: // Default (fall-through). // Get the key. for ( ; ; ) { switch (str[pos++]) { case '\0', ';': return e_INI_LINE_TYPE_INVALID; // No value. case '\\': if (str[pos]) end = ++pos; // Skip next character too. else return e_INI_LINE_TYPE_INVALID; // Escaping nothing. case '=': break; case '\1' .. ' ': {} // Whitespace, skip it. default: end = pos; // Characters, save this position. } } p0e = end; // See what comes next. INI_SkipWhitespace(str, pos); switch (str[pos]) { case '\0': return p1s = p1e = pos, e_INI_LINE_TYPE_ENTRY; case ';': { p1s = p1e = pos++, ret = e_INI_LINE_TYPE_ENTRY; goto state_in_comment; } case '\\': if (str[pos + 1]) p1s = pos++, end = ++pos; else return e_INI_LINE_TYPE_INVALID; default: p1s = pos++, end = pos; } for ( ; ; ) { switch (str[pos++]) { case '\0': return p1e = end, e_INI_LINE_TYPE_ENTRY; case ';': { p1e = end, ret = e_INI_LINE_TYPE_ENTRY; goto state_in_comment; } case '\\': if (str[pos]) end = ++pos; // Skip next character too. else return e_INI_LINE_TYPE_INVALID; // Escaping nothing. case '\1' .. ' ': {} // Whitespace, skip it. default: end = pos; // Characters, save this position. } } state_in_tag: // Get the tag name. INI_SkipWhitespace(str, pos); p0s = pos; for ( ; ; ) { switch (str[pos++]) { case '\0', ';': return e_INI_LINE_TYPE_INVALID; // No tag end. case '\\': if (str[pos]) end = ++pos; // Skip next character too. else return e_INI_LINE_TYPE_INVALID; // Escaping nothing. case ']': if (end) break; // End of the tag. else return e_INI_LINE_TYPE_INVALID; // Tag is empty. case '\1' .. ' ': {} // Whitespace, skip it. default: end = pos; // Characters, save this position. } } p0e = end; // See what comes next. INI_SkipWhitespace(str, pos); switch (str[pos++]) { case '\0': return e_INI_LINE_TYPE_TAG; // Line over. case ';': { // Skip over the comments. ret = e_INI_LINE_TYPE_TAG; goto state_in_comment; } case ':': {} default : return e_INI_LINE_TYPE_INVALID; // Unexpected characters. } // Get the inheritance. INI_SkipWhitespace(str, pos); if (!str[pos]) return e_INI_LINE_TYPE_INVALID; // No parent tag. p1s = pos; while (str[pos] > ' ') ++pos; p1e = pos; INI_SkipWhitespace(str, pos); switch (str[pos++]) { case '\0': return e_INI_LINE_TYPE_TAG; // Line over. case ';': ret = e_INI_LINE_TYPE_TAG; default : return e_INI_LINE_TYPE_INVALID; // Unexpected characters. } state_in_comment: INI_SkipWhitespace(str, pos); if (str[pos]) { p2s = pos, // Non-empty comment. pos = strlen(str); while (pos-- && str[pos] <= ' ') {} p2e = pos + 1; } return ret; } /* 88 88 88 ,d 88 88 88 88 88 8b,dPPYba, MM88MMM ,adPPYba, 8b,dPPYba, 8b,dPPYba, ,adPPYYba, 88 88 88P' `"8a 88 a8P_____88 88P' "Y8 88P' `"8a "" `Y8 88 88 88 88 88 8PP""""""" 88 88 88 ,adPPPPP88 88 88 88 88 88, "8b, ,aa 88 88 88 88, ,88 88 88 88 88 "Y888 `"Ybbd8"' 88 88 88 `"8bbdP"Y8 88 */ /*-------------------------------------------------------------------------*//** * The format destination. * The source format. * The file we are currently parsing. * The format parameter ordering. * * Generates a partial function name for processing callbacks. Includes the * filename and a placeholder for the tag name. This now takes extra * characters in to account and strips or converts bits: * * some/dir/file name.ext * * Becomes: * * file_name * * Before being formatted in to the specified remote format. The filename * also takes in to account "/" directory separators and "\\" ones on Windows. * * Because the majority of this function is concerned with formatting just part * of the function name correctly, it short-circuits if it detects that there * is no place for the function name to go. * * This is quite a complex function, but is only called once per file parse. * *//*------------------------------------------------------------------------**/ _Y_INI_STATIC stock INI_SetupCallbackName(fmat[32], const remoteFormat[], filename[], const bool:bFileFirst) { // Copy the basic format over. strcpy(fmat, remoteFormat); // Before any complex filename parsing, check if we actually need it. new fpos = strfind(remoteFormat, "%s"); if (!bFileFirst) fpos = strfind(remoteFormat, "%s", false, fpos + 1); // Is there a position in "remoteFormat" for "filename"? if (fpos == -1) return 0; new pos, prev = 0; // Isolate just the VALID name of the file, not the path or extension. while ((pos = strfind(filename, "/" , false, pos) + 1)) prev = pos; pos = prev; if (IsWindows()) { while ((pos = strfind(filename, "\\", false, pos) + 1)) prev = pos; pos = prev; } // Got the start of the name, now find the end. for (new ch; ; ) { // Get the extent of the valid function characters. for ( ; ; ) { ch = filename[pos]; if ('a' <= ch <= 'z' || 'A' <= ch <= 'Z' || '0' <= ch <= '9' || ch == '_' || ch == '@') ++pos; else break; } // Make the secondary format. filename[pos] = '\0'; if (bFileFirst) format(fmat, sizeof (fmat), fmat, filename[prev], "%s"); else format(fmat, sizeof (fmat), fmat, "%s", filename[prev]); filename[pos] = ch; // Add spaces as "_". P:7("INI_SetupCallbackName: '%02x' \"%s\" -> \"%s\" %d %d %d", ch, filename[prev], fmat, fpos, prev, pos); if (ch == ' ') { fpos += pos - prev, strins(fmat, "_%s", fpos++), prev = ++pos; P:7("INI_SetupCallbackName: \"%s\" -> \"%s\" %d %d %d", filename[prev], fmat, fpos, prev, pos); } else break; } return 1; } _Y_INI_STATIC stock INI_MakeCallbackFormat(const bool:bExtra, const bool:bPassTag, callbackFormat[5]) { if (bExtra) { if (bPassTag) callbackFormat = _F; else callbackFormat = _F; } else { if (bPassTag) callbackFormat = _F; else callbackFormat = _F; } } /*-------------------------------------------------------------------------*//** * The file to load. * The format string to generate the remote function * t pass the data to once loaded. * The order of the remoteFormat parameters. * Send additional data. * Additional data to send. * Call local functions instead of global ones. * Pass the tag as an extra parameter not the function * name. * Apply the tag name filter to all tags or just prefixed * ones? * Text to use to search for which tags to load. * * bFileFirst sets the order and inclusion of the possible remoteFormat * parameters. If true the format will add the filename first then the * current tag, if false the order will be reversed. This can also be used * to exclude one or the other from the function name by setting the required * parameter to be entered first and then only having one %s in the format * sting. The default order is tag first for languages compatibility. * *//*------------------------------------------------------------------------**/ static stock INI_DoParentTag(const epos, const search[], File:f, callback[E_CALLBACK_DATA], bool:bExtra, extra, bool:bPassTag, tag[], bool:bFilter, filter[]) { // The bulk of this function is basically the same as INI_ParseFile (which // is a shame as it is a symptom of poor, unmaintainable code). fseek(f, 0, seek_start); // Jump back to the start. P:4("bool:INI_DoParentTag called: \"%s\", %i, %i, %i, %i, \"%s\"", search, bExtra, extra, bPassTag, bFilter, filter); new ppos, bool:handle = false, p0s, p0e, p1s, p1e, p2s, p2e; while ((ppos += fread(f, YSI_g_sCurLine)) < epos) { switch (INI_IdentifyLineType(YSI_g_sCurLine, p0s, p0e, p1s, p1e, p2s, p2e)) { case e_INI_LINE_TYPE_INVALID, e_INI_LINE_TYPE_DATALESS: {} case e_INI_LINE_TYPE_TAG: { // A new tag. // First, check if it is a tag we might care about. YSI_g_sCurLine[p0e] = '\0'; if (YSI_g_sCurLine[p0s] == '@' && YSI_g_sCurLine[p0s + 1] == '@' && (p0e = strfind(YSI_g_sCurLine, "-")) != -1) { // Check if the current tag is one of the ones we want to // filer for. The "@@" prefix is the "filterable" prefix. // If there is no filter then everything will be loaded. if (p0e == p0s + 2 || !YSI_g_sCurLine[p0e + 1]) continue; YSI_g_sCurLine[p0e] = '\0'; if (bFilter && strcmp(YSI_g_sCurLine[p0s + 2], filter)) { // Only skip this if filtering is enabled. We can't put // the "bFilter" check higher or the "-" won't be found // to resolve the tag name accurately. continue; } p0s = p0e + 1; } if (!strcmp(YSI_g_sCurLine[p0s], search)) { handle = true; if (p1s != p1e) { new parent[32]; YSI_g_sCurLine[p1e] = '\0', strcat(parent, YSI_g_sCurLine[p1s]); if (!INI_DoParentTag(ppos, parent, f, callback, bExtra, extra, bPassTag, tag, bFilter, filter)) return false; fseek(f, ppos, seek_start); } } else if (handle) { // Parent tag over. return true; } } default: // e_INI_LINE_TYPE_ENTRY and e_INI_LINE_TYPE_CONT. { if (!handle) continue; // Standard key-value pair. YSI_g_sCurLine[p0e] = YSI_g_sCurLine[p1e] = '\0'; if (bExtra) { if (bPassTag) Callback_Call(callback, extra, tag, YSI_g_sCurLine[p0s], YSI_g_sCurLine[p1s]); else Callback_Call(callback, extra, YSI_g_sCurLine[p0s], YSI_g_sCurLine[p1s]); } else { if (bPassTag) Callback_Call(callback, tag, YSI_g_sCurLine[p0s], YSI_g_sCurLine[p1s]); else Callback_Call(callback, YSI_g_sCurLine[p0s], YSI_g_sCurLine[p1s]); } } } } // Return to where we were before. return handle; }