resource_control.cpp 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  1. /*************************************************************************
  2. * *
  3. * Open Dynamics Engine, Copyright (C) 2001,2002 Russell L. Smith. *
  4. * All rights reserved. Email: [email protected] Web: www.q12.org *
  5. * *
  6. * This library is free software; you can redistribute it and/or *
  7. * modify it under the terms of EITHER: *
  8. * (1) The GNU Lesser General Public License as published by the Free *
  9. * Software Foundation; either version 2.1 of the License, or (at *
  10. * your option) any later version. The text of the GNU Lesser *
  11. * General Public License is included with this library in the *
  12. * file LICENSE.TXT. *
  13. * (2) The BSD-style license that is included with this library in *
  14. * the file LICENSE-BSD.TXT. *
  15. * *
  16. * This library is distributed in the hope that it will be useful, *
  17. * but WITHOUT ANY WARRANTY; without even the implied warranty of *
  18. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the files *
  19. * LICENSE.TXT and LICENSE-BSD.TXT for more details. *
  20. * *
  21. *************************************************************************/
  22. /*
  23. * Resource accounting/preallocation class implementations
  24. * Copyright (c) 2017-2020 Oleh Derevenko, [email protected] (change all "a" to "e")
  25. */
  26. #include <ode/common.h>
  27. #include <ode/cooperative.h>
  28. #include "config.h"
  29. #include "resource_control.h"
  30. #include "simple_cooperative.h"
  31. //////////////////////////////////////////////////////////////////////////
  32. // dxResourceRequirementDescriptor();
  33. dxResourceRequirementDescriptor::~dxResourceRequirementDescriptor()
  34. {
  35. // Do nothing
  36. }
  37. //////////////////////////////////////////////////////////////////////////
  38. // dxRequiredResourceContainer
  39. dxRequiredResourceContainer::~dxRequiredResourceContainer()
  40. {
  41. freeResources();
  42. }
  43. bool dxRequiredResourceContainer::allocateResources(const dxResourceRequirementDescriptor &requirementDescriptor)
  44. {
  45. bool result = false;
  46. bool bufferAllocated = false;
  47. do
  48. {
  49. sizeint memorySizeRequirement = requirementDescriptor.getMemorySizeRequirement();
  50. if (memorySizeRequirement != 0)
  51. {
  52. unsigned memoryAlignmentRequirement = requirementDescriptor.getMemoryAlignmentRequirement();
  53. void *bufferAllocated = m_memoryAllocation.allocAligned(memorySizeRequirement, memoryAlignmentRequirement);
  54. if (bufferAllocated == NULL)
  55. {
  56. break;
  57. }
  58. }
  59. bufferAllocated = true;
  60. dxThreadingBase *relatedThreading = requirementDescriptor.getrelatedThreading();
  61. dIASSERT(relatedThreading != NULL);
  62. unsigned simultaneousCallRequirement = requirementDescriptor.getSimultaneousCallRequirement();
  63. if (simultaneousCallRequirement != 0)
  64. {
  65. if (!relatedThreading->PreallocateResourcesForThreadedCalls(simultaneousCallRequirement))
  66. {
  67. break;
  68. }
  69. }
  70. dCallWaitID stockCallWait = NULL;
  71. if (requirementDescriptor.getIsStockCallWaitRequired())
  72. {
  73. stockCallWait = relatedThreading->AllocateOrRetrieveStockCallWaitID();
  74. if (stockCallWait == NULL)
  75. {
  76. break;
  77. }
  78. }
  79. m_relatedThreading = relatedThreading;
  80. m_stockCallWait = stockCallWait;
  81. result = true;
  82. }
  83. while (false);
  84. if (!result)
  85. {
  86. if (bufferAllocated)
  87. {
  88. m_memoryAllocation.freeAllocation();
  89. }
  90. }
  91. return result;
  92. }
  93. void dxRequiredResourceContainer::freeResources()
  94. {
  95. if (m_relatedThreading != NULL)
  96. {
  97. m_relatedThreading = NULL;
  98. m_stockCallWait = NULL;
  99. m_memoryAllocation.freeAllocation();
  100. }
  101. else
  102. {
  103. dIASSERT(m_stockCallWait == NULL);
  104. dIASSERT(m_memoryAllocation.getUserAreaPointer() == NULL);
  105. }
  106. }
  107. //////////////////////////////////////////////////////////////////////////
  108. // Public interface functions
  109. static inline
  110. dResourceRequirementsID encodeResourceRequirementsID(dxResourceRequirementDescriptor *requirementsDescriptor)
  111. {
  112. return (dResourceRequirementsID)requirementsDescriptor;
  113. }
  114. /*extern ODE_API */
  115. dResourceRequirementsID dResourceRequirementsCreate(dCooperativeID cooperative)
  116. {
  117. dAASSERT(cooperative != NULL);
  118. dxSimpleCooperative *cooperativeInstance = decodeCooperativeID(cooperative);
  119. dxThreadingBase *threading = cooperativeInstance->getRelatedThreading();
  120. dxResourceRequirementDescriptor *requirementsDescriptor = new dxResourceRequirementDescriptor(threading);
  121. dResourceRequirementsID result = encodeResourceRequirementsID(requirementsDescriptor);
  122. return result;
  123. }
  124. /*extern ODE_API */
  125. void dResourceRequirementsDestroy(dResourceRequirementsID requirements)
  126. {
  127. dxResourceRequirementDescriptor *requirementsDescriptor = decodeResourceRequirementsID(requirements);
  128. if (requirementsDescriptor != NULL)
  129. {
  130. delete requirementsDescriptor;
  131. }
  132. }
  133. /*extern ODE_API */
  134. dResourceRequirementsID dResourceRequirementsClone(/*const */dResourceRequirementsID requirements)
  135. {
  136. dAASSERT(requirements != NULL);
  137. dxResourceRequirementDescriptor *requirementsDescriptor = decodeResourceRequirementsID(requirements);
  138. dxResourceRequirementDescriptor *descriptorClone = new dxResourceRequirementDescriptor(*requirementsDescriptor);
  139. dResourceRequirementsID result = encodeResourceRequirementsID(descriptorClone);
  140. return result;
  141. }
  142. /*extern ODE_API */
  143. void dResourceRequirementsMergeIn(dResourceRequirementsID summaryRequirements, /*const */dResourceRequirementsID extraRequirements)
  144. {
  145. dAASSERT(summaryRequirements != NULL);
  146. dAASSERT(extraRequirements != NULL);
  147. dxResourceRequirementDescriptor *summaryDescriptor = decodeResourceRequirementsID(summaryRequirements);
  148. dxResourceRequirementDescriptor *extraDescriptor = decodeResourceRequirementsID(extraRequirements);
  149. summaryDescriptor->mergeAnotherDescriptorIn(*extraDescriptor);
  150. }
  151. //////////////////////////////////////////////////////////////////////////
  152. static inline
  153. dResourceContainerID encodeResourceContainerID(dxRequiredResourceContainer *containerInstance)
  154. {
  155. return (dResourceContainerID)containerInstance;
  156. }
  157. /*extern ODE_API */
  158. dResourceContainerID dResourceContainerAcquire(/*const */dResourceRequirementsID requirements)
  159. {
  160. dAASSERT(requirements != NULL);
  161. dResourceContainerID result = NULL;
  162. bool allocationSucceeded = false;
  163. dxRequiredResourceContainer *containerInstance;
  164. bool containerAllocated = false;
  165. dxResourceRequirementDescriptor *requirementsInstance = decodeResourceRequirementsID(requirements);
  166. do
  167. {
  168. containerInstance = new dxRequiredResourceContainer();
  169. if (containerInstance == NULL)
  170. {
  171. break;
  172. }
  173. containerAllocated = true;
  174. if (!containerInstance->allocateResources(*requirementsInstance))
  175. {
  176. break;
  177. }
  178. result = encodeResourceContainerID(containerInstance);
  179. allocationSucceeded = true;
  180. }
  181. while (false);
  182. if (!allocationSucceeded)
  183. {
  184. if (containerAllocated)
  185. {
  186. delete containerInstance;
  187. }
  188. }
  189. return result;
  190. }
  191. /*extern ODE_API */
  192. void dResourceContainerDestroy(dResourceContainerID resources)
  193. {
  194. dxRequiredResourceContainer *containerInstance = decodeResourceContainerID(resources);
  195. if (containerInstance != NULL)
  196. {
  197. delete containerInstance;
  198. }
  199. }