Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

176 lines
4.3 KiB

  1. /*++
  2. Copyright (c) 1992 Microsoft Corporation
  3. Module Name:
  4. packstr.c
  5. Abstract:
  6. Contains functions for packing strings into buffers that also contain
  7. structures.
  8. Author:
  9. Dan Lafferty (danl) 13-Jan-1992
  10. Environment:
  11. User Mode -Win32
  12. Revision History:
  13. 13-Jan-1992 danl
  14. stole from netlib.
  15. 27-May-1992 JohnRo
  16. RAID 9829: winsvc.h and related file cleanup.
  17. Fixed a UNICODE-related bug.
  18. --*/
  19. #include <scpragma.h>
  20. #include <windef.h> // IN, OUT, etc.
  21. #include <sclib.h> // My prototypes.
  22. #include <string.h> // strncpy
  23. #include <stdlib.h> // wcsncpy
  24. BOOL
  25. ScCopyStringToBufferW (
  26. IN LPCWSTR String OPTIONAL,
  27. IN DWORD CharacterCount,
  28. IN LPCWSTR FixedDataEnd,
  29. IN OUT LPWSTR *EndOfVariableData,
  30. OUT LPWSTR *VariableDataPointer,
  31. IN const LPBYTE lpBufferStart OPTIONAL
  32. )
  33. /*++
  34. Routine Description:
  35. This routine puts a single variable-length string into an output buffer.
  36. The string is not written if it would overwrite the last fixed structure
  37. in the buffer.
  38. The code is swiped from svcsupp.c written by DavidTr.
  39. Sample usage:
  40. LPBYTE FixedDataEnd = OutputBuffer + sizeof(WKSTA_INFO_202);
  41. LPWSTR EndOfVariableData = OutputBuffer + OutputBufferSize;
  42. //
  43. // Copy user name
  44. //
  45. ScCopyStringToBuffer(
  46. UserInfo->UserName.Buffer;
  47. UserInfo->UserName.Length;
  48. FixedDataEnd,
  49. &EndOfVariableData,
  50. &WkstaInfo->wki202_username,
  51. NULL
  52. );
  53. Arguments:
  54. String - Supplies a pointer to the source string to copy into the
  55. output buffer. If String is null then a pointer to a zero terminator
  56. is inserted into output buffer.
  57. CharacterCount - Supplies the length of String, not including zero
  58. terminator. (This in units of characters - not bytes).
  59. FixedDataEnd - Supplies a pointer to just after the end of the last
  60. fixed structure in the buffer.
  61. EndOfVariableData - Supplies an address to a pointer to just after the
  62. last position in the output buffer that variable data can occupy.
  63. Returns a pointer to the string written in the output buffer.
  64. VariableDataPointer - Supplies a pointer to the place in the fixed
  65. portion of the output buffer where a pointer or offset to the
  66. variable data should be written. Pointer vs. offset is decided
  67. based on the presence of lpBufferStart
  68. lpBufferStart - If NULL, VariableDataPointer should be filled with
  69. a pointer to the string written into the buffer. If non-NULL,
  70. VariableDataPointer should be filled with an offset to the
  71. string written into the buffer.
  72. Return Value:
  73. Returns TRUE if string fits into output buffer, FALSE otherwise.
  74. --*/
  75. {
  76. DWORD CharsNeeded = (CharacterCount + 1);
  77. //
  78. // Determine if string will fit, allowing for a zero terminator. If no,
  79. // just set the pointer to NULL.
  80. //
  81. if ((*EndOfVariableData - CharsNeeded) >= FixedDataEnd) {
  82. //
  83. // It fits. Move EndOfVariableData pointer up to the location where
  84. // we will write the string.
  85. //
  86. *EndOfVariableData -= CharsNeeded;
  87. //
  88. // Copy the string to the buffer if it is not null.
  89. //
  90. if (CharacterCount > 0 && String != NULL) {
  91. wcsncpy(*EndOfVariableData, String, CharacterCount);
  92. }
  93. //
  94. // Set the zero terminator.
  95. //
  96. *(*EndOfVariableData + CharacterCount) = L'\0';
  97. //
  98. // Set up the pointer in the fixed data portion to point to where the
  99. // string is written.
  100. //
  101. if (lpBufferStart != NULL)
  102. {
  103. *(LPDWORD) VariableDataPointer =
  104. (DWORD) ((LPBYTE) *EndOfVariableData - lpBufferStart);
  105. }
  106. else
  107. {
  108. *VariableDataPointer = *EndOfVariableData;
  109. }
  110. return TRUE;
  111. }
  112. else {
  113. //
  114. // It doesn't fit. Set the offset to NULL.
  115. //
  116. if (lpBufferStart != NULL)
  117. {
  118. *(LPDWORD) VariableDataPointer = 0;
  119. }
  120. else
  121. {
  122. *VariableDataPointer = NULL;
  123. }
  124. return FALSE;
  125. }
  126. } // ScCopyStringToBuffer