codec.c
上传用户:wstnjxml
上传日期:2014-04-03
资源大小:7248k
文件大小:6k
源码类别:

Windows CE

开发平台:

C/C++

  1. /*****************************************************************************
  2.  *
  3.  * This program is free software ; you can redistribute it and/or modify
  4.  * it under the terms of the GNU General Public License as published by
  5.  * the Free Software Foundation; either version 2 of the License, or
  6.  * (at your option) any later version.
  7.  *
  8.  * This program is distributed in the hope that it will be useful,
  9.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11.  * GNU General Public License for more details.
  12.  *
  13.  * You should have received a copy of the GNU General Public License
  14.  * along with this program; if not, write to the Free Software
  15.  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
  16.  *
  17.  * $Id: codec.c 438 2005-12-29 17:23:54Z picard $
  18.  *
  19.  * The Core Pocket Media Player
  20.  * Copyright (c) 2004-2005 Gabor Kovacs
  21.  *
  22.  ****************************************************************************/
  23. #include "common.h"
  24. static const datatable CodecParams[] = 
  25. {
  26. { CODEC_INPUT, TYPE_PACKET, DF_INPUT },
  27. { CODEC_OUTPUT, TYPE_PACKET, DF_OUTPUT },
  28. { CODEC_COMMENT, TYPE_COMMENT, DF_OUTPUT },
  29. DATATABLE_END(CODEC_CLASS)
  30. };
  31. static void ClearInput(codec* p)
  32. {
  33. PacketFormatClear(&p->In.Format);
  34. PacketFormatClear(&p->Out.Format);
  35. p->Out.Process = DummyProcess;
  36. p->In.Process = DummyProcess;
  37. }
  38. int CodecEnum(codec* p, int* No, datadef* Param)
  39. {
  40. if (FlowEnum(p,No,Param)==ERR_NONE)
  41. return ERR_NONE;
  42. if (NodeEnumTable(No,Param,CodecParams)==ERR_NONE)
  43. {
  44. if (!p->UseComment && Param->No==CODEC_COMMENT)
  45. Param->Flags |= DF_RDONLY;
  46. return ERR_NONE;
  47. }
  48. return ERR_INVALID_PARAM;
  49. }
  50. int CodecGet(codec* p, int No, void* Data, int Size)
  51. {
  52. int Result = ERR_INVALID_PARAM;
  53. switch (No)
  54. {
  55. case CODEC_INPUT: GETVALUE(p->In.Pin,pin); break;
  56. case CODEC_INPUT|PIN_FORMAT: GETVALUE(p->In.Format,packetformat); break;
  57. case CODEC_INPUT|PIN_PROCESS: GETVALUE(p->In.Process,packetprocess); break;
  58. case CODEC_OUTPUT: GETVALUE(p->Out.Pin,pin); break;
  59. case CODEC_OUTPUT|PIN_FORMAT: GETVALUE(p->Out.Format,packetformat); break;
  60. case CODEC_OUTPUT|PIN_PROCESS: GETVALUE(p->Out.Process,packetprocess); break;
  61. case CODEC_COMMENT: GETVALUE(p->Comment,pin); break;
  62. case FLOW_BUFFERED: GETVALUE(p->ReSend!=NULL,bool_t); break;
  63. }
  64. return Result;
  65. }
  66. static int Flush(codec* p)
  67. {
  68. int Result = ERR_NONE;
  69. p->Pending = 0;
  70. p->Packet.RefTime = TIME_UNKNOWN;
  71. if (p->Flush)
  72. p->Flush(p);
  73. return Result;
  74. }
  75. static int Process(codec* p, const packet* Packet, const flowstate* State)
  76. {
  77. flowstate State2;
  78. int Result;
  79. if (p->Pending)
  80. {
  81. do
  82. {
  83. Result = p->Out.Process(p->Out.Pin.Node,&p->Packet,State);
  84. if (Result == ERR_BUFFER_FULL)
  85. return Result;
  86. }
  87. while (p->Process(p,NULL,NULL) == ERR_NONE);
  88. p->Pending = 0;
  89. }
  90. if (State->DropLevel > 1 && !p->NoHardDropping)
  91. {
  92. Flush(p);
  93. p->Out.Process(p->Out.Pin.Node,NULL,State); // signal dropped packet
  94. return ERR_NONE;
  95. }
  96. Result = p->Process(p,Packet,State);
  97. if (Result == CODEC_NO_DROPPING)
  98. {
  99. State2.CurrTime = State->CurrTime;
  100. State2.DropLevel = 0;
  101. State = &State2;
  102. Result = ERR_NONE;
  103. }
  104. if (Result != ERR_NONE)
  105. {
  106. if (!Packet)
  107. return p->Out.Process(p->Out.Pin.Node,NULL,State);
  108. return Result;
  109. }
  110. do
  111. {
  112. int n = p->Out.Process(p->Out.Pin.Node,&p->Packet,State);
  113. if (n == ERR_BUFFER_FULL)
  114. {
  115. p->Pending = 1; 
  116. if (!Packet) //return previous result if Packet already processed
  117. Result = n;
  118. break;
  119. }
  120. Result = n;
  121. if (Result == ERR_NONE && State->CurrTime == TIME_SYNC)
  122. {
  123. p->Packet.RefTime = TIME_UNKNOWN;
  124. break;
  125. }
  126. }
  127. while (p->Process(p,NULL,NULL) == ERR_NONE);
  128. return Result;
  129. }
  130. static int UpdateInput(codec* p)
  131. {
  132. int Result2;
  133. int Result = ERR_NONE;
  134. Flush(p);
  135. if (p->In.Format.Type != PACKET_NONE)
  136. {
  137. if (!PacketFormatMatch(p->Node.Class, &p->In.Format))
  138. {
  139. ClearInput(p);
  140. Result = ERR_NOT_SUPPORTED;
  141. }
  142. if (p->Process)
  143. p->In.Process = (packetprocess)Process; // using default codec process
  144. }
  145. if (p->UpdateInput)
  146. {
  147. Result2 = p->UpdateInput(p);
  148. if (Result == ERR_NONE)
  149. Result = Result2;
  150. }
  151. if (Result != ERR_NONE)
  152. ClearInput(p);
  153. Result2 = ConnectionUpdate(&p->Node,CODEC_OUTPUT,p->Out.Pin.Node,p->Out.Pin.No);
  154. if (Result == ERR_NONE)
  155. Result = Result2;
  156. return Result;
  157. }
  158. int CodecSet(codec* p, int No, const void* Data, int Size)
  159. {
  160. int Result = ERR_INVALID_PARAM;
  161. switch (No)
  162. {
  163. case CODEC_INPUT: SETVALUE(p->In.Pin,pin,ERR_NONE); break;
  164. case CODEC_INPUT|PIN_FORMAT: SETPACKETFORMAT(p->In.Format,packetformat,UpdateInput(p)); break;
  165. case CODEC_OUTPUT: SETVALUE(p->Out.Pin,pin,ERR_NONE); break;
  166. case CODEC_OUTPUT|PIN_FORMAT: if (p->UpdateOutput) SETPACKETFORMAT(p->In.Format,packetformat,p->UpdateOutput(p)); break;
  167. case CODEC_OUTPUT|PIN_PROCESS: SETVALUE(p->Out.Process,packetprocess,ERR_NONE); break;
  168. case CODEC_COMMENT: SETVALUE(p->Comment,pin,ERR_NONE); break;
  169. case FLOW_FLUSH: Result = Flush(p); break;
  170. case FLOW_RESEND: if (p->ReSend) Result = p->ReSend(p); break;
  171. case FLOW_NOT_SUPPORTED: SETVALUE(p->NotSupported,pin,ERR_NONE); break;
  172. }
  173. return Result;
  174. }
  175. static int Create(codec* p)
  176. {
  177. p->Node.Enum = (nodeenum)CodecEnum;
  178. p->Node.Get = (nodeget)CodecGet;
  179. p->Node.Set = (nodeset)CodecSet;
  180. ClearInput(p);
  181. return ERR_NONE;
  182. }
  183. static void Delete(codec* p)
  184. {
  185. ClearInput(p);
  186. if (p->UpdateInput)
  187. p->UpdateInput(p);
  188. }
  189. static const nodedef Codec =
  190. {
  191. sizeof(codec)|CF_ABSTRACT,
  192. CODEC_CLASS,
  193. FLOW_CLASS,
  194. PRI_DEFAULT,
  195. (nodecreate)Create,
  196. (nodedelete)Delete,
  197. };
  198. void Codec_Init()
  199. {
  200. NodeRegisterClass(&Codec);
  201. }
  202. void Codec_Done()
  203. {
  204. NodeUnRegisterClass(CODEC_CLASS);
  205. }