
    hh                     6   d Z ddlZddlZddlZddlmZ ddlZddlZddl	m
Z ddl	mZ ddlmZ ddlmZ ddlmZ dd	lmZ dd
lmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlm Z  ddlm!Z" ddl#m$Z$ ddl%m&Z' dZ(ejR                  Z)ejT                  Z*ejV                  Z+	 ddl,m-Z. e j`                  e jb                  e jd                  e jf                  e jh                  e jj                  e jl                  e jn                  e jp                  e jr                  e jt                  e jv                  e jx                  dZ=dZ>e jr                  e jr                  e jf                  he jn                  e jn                  hiZ?d Z@d ZAd ZBd ZCd ZDd ZE	 d8dZFd ZGd ZHd  ZId! ZJd" ZKd# ZL	 	 	 	 d9d$ZMd% ZNd& ZOd' ZPd( ZQd) ZRd* ZSe j`                  fd+ZTd, ZUe j`                  fd-ZVd. ZWd/ ZXd0 ZYe j`                  e j`                  fd1ZZd2 Z[d3 Z\d4 Z]d5 Z^d6 Z_d7 Z`y# e/$ r dZ.Y sw xY w):z+Functions used by multiple converter files.    N)logging)
config_pb2)meta_graph_pb2)'conversion_metadata_schema_py_generated)schema_py_generated)schema_util)tflite_keras_util)convert_op_hints_to_stubs)find_all_hinted_output_nodes)flatbuffer_utils)function)convert_to_constants)dtypes)error_interpolation)tf_optimizer)export_meta_graphCONVERSION_METADATA)jit)r                           	   
         s   TFL3c                 r    t         j                  |       }|t        dj                  | t                     |S )a  Converts tflite enum type (eg: 0) to tf type (eg: tf.float32).

  Args:
    tflite_enum_type: tflite enum type (eg: 0, that corresponds to float32)

  Raises:
    ValueError: If an invalid tflite enum type is provided.

  Returns:
    tf type (eg: tf.float32)
  z>Unsupported enum {}. The valid map of enum to tf types is : {})_MAP_TFLITE_ENUM_TO_TF_TYPESget
ValueErrorformat)tflite_enum_typetf_types     Y/var/www/html/dev/engine/venv/lib/python3.12/site-packages/tensorflow/lite/python/util.py$_convert_tflite_enum_type_to_tf_typer)   T   sA     ),,-=>'_
H	 ">	?A A 
.    c                 (    | rd| j                   z   S dS )z=Converts tf.dtype (eg: tf.float32) to str (eg: "tf.float32").ztf.N)name)r'   s    r(   get_tf_type_namer-   h   s    !(	2d2r*   c                     | j                   j                  d      }t        |      dkD  r&t        dj	                  t        |      dz
              t        |      dkD  r|d   dk7  r| j                   S |d   S )zWReturns name of the input tensor.

  Args:
    tensor: tf.Tensor

  Returns:
    str
  :r   z1Tensor name invalid. Expect 0 or 1 colon, got {0}r   0r   )r,   splitlenr$   r%   )tensorpartss     r(   get_tensor_namer5   m   ss     ++

C
 %Z!^
HOOE
Q  
 	Z!^aC;;	q/r*   c                    i }| j                         D ]%  }|j                         D ]  }||t        |      <    ' g }g }|D ]l  }t        |t              s$t        dj                  t        |      |            |j                  |      }||j                  |       \|j                  |       n |r)t        dj                  dj                  |                  |S )ae  Gets the Tensors associated with the `tensor_names` in the provided graph.

  Args:
    graph: TensorFlow Graph.
    tensor_names: List of strings that represent names of tensors in the graph.

  Returns:
    A list of Tensor objects in the same order the names are provided.

  Raises:
    ValueError:
      tensor_names contains an invalid tensor name.
  zInvalid type for a tensor name in the provided graph. Expected type for a tensor name is 'str', instead got type '{}' for tensor name '{}'z Invalid tensors '{}' were found.,)get_operationsvaluesr5   
isinstancestrr$   r%   typer#   appendjoin)graphtensor_namestensor_name_to_tensoropr3   tensorsinvalid_tensorsr,   s           r(   get_tensors_from_tensor_namesrE      s       " >b))+ >7=OF34>>
 '/ ddC  88>:t9-. .
 #&&t,F~T"nnV 
7>>!# $ $	.r*   c           
      l   |rj| D ci c]  }t        |      | }}|j                         D ]=  \  }}||vrt        dj                  |            |'||   }	 |j	                  |       ? yyc c}w # t        $ r7}dj                  ||j
                  |t        |            }t        |      d}~ww xY w)a  Sets Tensor shape for each tensor if the shape is defined.

  Args:
    tensors: TensorFlow tensor.Tensor.
    shapes: Dict of strings representing input tensor names to list of
      integers representing input shapes (e.g., {"foo": : [1, 16, 16, 3]}).

  Raises:
    ValueError:
      `shapes` contains an invalid tensor.
      `shapes` contains an invalid shape for a valid tensor.
  z/Invalid tensor '{}' found in tensor shapes map.Nz@The shape of tensor '{0}' cannot be changed from {1} to {2}. {3})r5   itemsr$   r%   	set_shapeshaper;   )rC   shapesr3   tensor_names_to_tensorr,   rI   errormessages           r(   set_tensor_shapesrN      s     6=,2'  ||~ $e	+	+   &t. 	.		'-	$


5
!$	   	$  &tV\\5#e* M 7#
#	$s   A.A33	B3<2B..B3c                     t        j                         }|j                  j                  }| D ]  }|j                  j                  |        |S )zCreates a tf.compat.v1.ConfigProto for configuring Grappler.

  Args:
    optimizers_list: List of strings that represents the list of optimizers.

  Returns:
    tf.ConfigProto.
  )_config_pb2ConfigProtograph_optionsrewrite_options
optimizersr=   )optimizers_listconfigrS   	optimizers       r(   get_grappler_configrX      sK     ""$&((88/" 1i%%i01	-r*   c                 .   t        | |      }t        j                         }|D ]  }|j                  |j                  |j                     _        |j
                  j                  |j                  |j                     _        |j                  |j                     j                  j                  |j                  j                                 |D ]  }|j                  |j                  |j                     _        |j
                  j                  |j                  |j                     _        |j                  |j                     j                  j                  |j                  j                                 |j                  d   j                  |       t        j                         }||z   D ]1  }|j                  j                  j!                  |j                         3 |j"                  d   j                  |       t%        j&                  ||      S )a  Apply standard TensorFlow optimizations to the graph_def.

  Args:
    graph_def: Frozen GraphDef to be optimized.
    input_arrays: List of arrays that are considered inputs of the graph.
    output_arrays: List of arrays that are considered outputs of the graph.
    config: tf.ConfigProto.
    graph: TensorFlow Graph. Required when Eager mode is enabled. (default None)

  Returns:
    A new, optimized GraphDef.
  )	graph_defr?   not_used_keytrain_op)_export_meta_graph_meta_graph_pb2SignatureDefr,   inputsdtypeas_datatype_enumtensor_shapeCopyFromrI   as_protooutputssignature_defCollectionDef	node_listvaluer=   collection_defr   OptimizeGraph)	rZ   input_arraysoutput_arraysrV   r?   
meta_graph	signaturearrayfetch_collections	            r(   run_graph_optimizationsrs      s   " "IUC***,) Oe(-

IUZZ %).)E)EIUZZ &UZZ --66u{{7K7K7MNO
  Pe).Iejj!&*/++*F*FIejj!'ejj!..778L8L8NOP
 >*33I> %224m+ 8e$$++EJJ78J'001AB		#	#FJ	77r*   c                     t        |       rt        d      |D cg c]  }t        |       }}t        j                  | |||z         }t        |      }|S c c}w )Nz.Try to convert op hints, needs unfrozen graph.)rZ   )is_frozen_graphr$   r5   _convert_to_constantsconvert_variables_to_constantsr
   )sessrZ   output_tensorshinted_outputs_nodesr3   rn   s         r(   _convert_op_hints_if_presentr{     sa    T
E
FF9GHv?6*H-H#BB
I}';;=)')<)		 Is   Ac                 ~   t        j                  | j                        }t        dg      }t	        ||||| j
                        }t        |       }|rt        | |||      S t        |       sB|D cg c]   }|j                  j                  d      d   " }}t        j                  | ||      S | j                  S c c}w )a  Returns a frozen GraphDef.

  Runs a Grappler pass and freezes a graph with Variables in it. Otherwise the
  existing GraphDef is returned. The Grappler pass is only run on models that
  are frozen in order to inline the functions in the graph.
  If OpHints is present, it will try to convert the OpHint graph.

  Args:
    sess: TensorFlow Session.
    input_tensors: List of input tensors.
    output_tensors: List of output tensors (only .name is used from this).

  Returns:
    Frozen GraphDef.
  r   )r?   r/   r   )rv    disable_lower_using_switch_mergerZ   rX   rs   r?   r   r{   ru   r,   r1   rw   )rx   input_tensorsry   rZ   rV   rz   r3   output_node_namess           r(   freeze_graphr     s    ( $DD
nn)
|,&%djjJ) 6d;'i(<> > 
	AOPv**3/2PP ??i*  >> Qs   0%B:c                     | j                   j                         D ]:  }|j                  j                  d      s|j                  j	                  d      s: y y)zDetermines if the graph is frozen.

  Determines if a graph has previously been frozen by checking for any
  operations of type Variable*. If variables are found, the graph is not frozen.

  Args:
    sess: TensorFlow Session.

  Returns:
    Bool.
  Variable
VariableOpFT)r?   r8   r<   
startswithendswith)rx   rB   s     r(   ru   ru   :  sK     JJ%%' b	ww*%)9)9,)G 
r*   c                       fd}|S )a=  Returns a method to retrieve the `GraphDebugInfo` from the original graph.

  Args:
    original_graph: The original `Graph` containing all the op stack traces.

  Returns:
    A function which retrieves the stack traces from the original graph and
    converts them to a `GraphDebugInfo` for a given set of nodes.
  c                    syg }| D ]  \  }}	 |s#|j                  |j                  |      f       nxj                  |      }t        |t        j
                        r-|j                  ||j                  j                  |      f       n t        j                  j                  d        t        j                  |      S # t        $ r Y w xY w)CFunction to create `GraphDebugInfo` for the given `original_nodes`.Nz9Use '@tf.function' or '@defun' to decorate the function.
)r=   get_operation_by_name_get_functionr:   r   AtomicFunctionr?   sysstderrwriteKeyError_error_interpolationcreate_graph_debug_info_def)original_nodes
useful_opsfuncr,   sub_funcoriginal_graphs        r(   fz build_debug_info_func.<locals>.fW  s    J$ 
d


T>#G#G#MN
O#11$7(("9"9:x~~;;DABD JJLN"  ;;JGG	  s   BC	CC )r   r   s   ` r(   build_debug_info_funcr   L  s    H2 
(r*   c                       fd}|S )a:  Returns a method to retrieve the `GraphDebugInfo` from the original graph.

  Args:
    saved_debug_info: The `GraphDebugInfo` containing all the debug info.

  Returns:
    A function which retrieves the stack traces from the original graph and
    converts them to a `GraphDebugInfo` for a given set of nodes.
  c                 
    ~ S )r   r   )r   saved_debug_infos    r(   r   z"convert_debug_info_func.<locals>.f~  s    r*   r   )r   r   s   ` r(   convert_debug_info_funcr   s  s    
 
(r*   c                 n   | syt               }|j                  D ]  }|j                  j                  }|j                  j                  }|s|j                  d|j                  f       Ot        t        |            D ]-  }|t        |      k\  rdn||   }|j                  |||   f       /   | |      S )aI  Returns the debug info for the original nodes in the `converted_graph`.

  Args:
    nodes_to_debug_info_func: The method to collect the op debug info for the
      nodes.
    converted_graph: A `GraphDef` after optimization and transformation.

  Returns:
    `GraphDebugInfo` for all the original nodes in `converted_graph`.
  N )	setnodeexperimental_debug_infooriginal_node_namesoriginal_func_namesaddr,   ranger2   )nodes_to_debug_info_funcconverted_graphr   r   debug_nodesdebug_funcsi
debug_funcs           r(   get_debug_infor     s     
" 5."" 	9d..BBK..BBK"dii)S%& 9!K 00Rk!n
JA789	9 
".	11r*   c                 T   d}g }|}t        |       D ]2  }	t        |      dz   |kD  r|j                  |dz          |}|d|	fz  z  }4 t        |      t        |      kD  r|j                  |dz          dj                  |      }
|d|j	                         z   dz   }|dj                  |	      }nd}|r9d
j                  t        j                  j                         j                        }nd}d}|j                  |t        |       |
||      }d}|j                  |||      }||fS )a  Returns strings representing a C constant array containing `data`.

  Args:
    data: Byte array that will be converted into a C constant.
    array_name: String to use as the variable name for the constant array.
    max_line_width: The longest line length, for formatting purposes.
    include_guard: Name to use for the include guard macro definition.
    include_path: Optional path to include in the source file.
    use_tensorflow_license: Whether to include the standard TensorFlow Apache2
      license in the generated files.

  Returns:
    Text that can be compiled as a C source file to link in the data as a
    literal array of values.
    Text that can be used as a C header file to reference the literal array.
  z   r   
z 0x%02x,r   TENSORFLOW_LITE_UTIL__DATA_H_z#include "{include_path}"
)include_patha  
/* Copyright {year} The TensorFlow Authors. All Rights Reserved.

Licensed under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
==============================================================================*/
)yeara  {license_text}
// This is a TensorFlow Lite model file that has been converted into a C data
// array using the tensorflow.lite.util.convert_bytes_to_c_source() function.
// This form is useful for compiling into a binary for devices that don't have a
// file system.

{include_line}
// We need to keep the data array aligned on some architectures.
#ifdef __has_attribute
#define HAVE_ATTRIBUTE(x) __has_attribute(x)
#else
#define HAVE_ATTRIBUTE(x) 0
#endif
#if HAVE_ATTRIBUTE(aligned) || (defined(__GNUC__) && !defined(__clang__))
#define DATA_ALIGN_ATTRIBUTE __attribute__((aligned(4)))
#else
#define DATA_ALIGN_ATTRIBUTE
#endif

const unsigned char {array_name}[] DATA_ALIGN_ATTRIBUTE = {{
{array_values}}};
const int {array_name}_len = {array_length};
)
array_namearray_lengtharray_valueslicense_textinclude_linea  
{license_text}

// This is a TensorFlow Lite model file that has been converted into a C data
// array using the tensorflow.lite.util.convert_bytes_to_c_source() function.
// This form is useful for compiling into a binary for devices that don't have a
// file system.

#ifndef {include_guard}
#define {include_guard}

extern const unsigned char {array_name}[];
extern const int {array_name}_len;

#endif  // {include_guard}
)r   include_guardr   )
	bytearrayr2   r=   r>   upperr%   datetimedatetodayr   )datar   max_line_widthr   r   use_tensorflow_licensestarting_padarray_lines
array_linerj   r   r   r   source_templatesource_textheader_templateheader_texts                    r(   convert_bytes_to_c_sourcer     sq   . ,+* (eJ!~-d*+j*x''J	(
 	_s<((zD()%,+j.>.>.@@:MM299! : #L L F##%**F+ " L/0  &&t9 ' !+/"  &&! ' !+
 
k	!!r*   c                     t         j                  j                  | d      }t         j                  j	                  |      }t        j                  |      }|S )z@Converts a tflite model from a bytearray into a parsable object.r   )	schema_fbModelGetRootAsModelModelTInitFromObjcopydeepcopy)model_bytearraymodel_objects     r(   '_convert_model_from_bytearray_to_objectr     sB    //C,!!--l;,|,,	r*   c                     t        j                  d      }| j                  |      }|j                  |t               t        |j                               S )z@Converts a tflite model from a parsable object into a bytearray.i   )file_identifier)flatbuffersBuilderPackFinish_TFLITE_FILE_IDENTIFIERbytesOutput)r   buildermodel_offsets      r(   '_convert_model_from_object_to_bytearrayr   %  sG     %'""7+,	../F.G	w~~	  r*   c                     g }t        | j                        D ]I  \  }}t        j                  |      }|t        j
                  j                  k(  s9|j                  |       K |S zReturns the quantize op idx.)	enumerateoperatorCodesr   #get_builtin_code_from_operator_coder   BuiltinOperatorQUANTIZEr=   modelquant_opcode_idxsidxopcodebuiltin_codes        r(   get_quantize_opcode_idxr   .  s`    u223 $kc6BB6JLy00999s#$ 
r*   c                     g }t        | j                        D ]I  \  }}t        j                  |      }|t        j
                  j                  k(  s9|j                  |       K |S r   )r   r   r   r   r   r   
DEQUANTIZEr=   r   s        r(   get_dequantize_opcode_idxr   8  s`    u223 $kc6BB6JLy00;;;s#$ 
r*   c                     t        t        |             D ].  }| |   j                  |v s|| |   j                     | |   _        0 y)z4Update the tensors in the SignatureDef's TensorMaps.N)r   r2   tensorIndex)tensor_mapsmap_old_to_new_tensorsr   s      r(   _update_signature_def_tensorsr   B  sL    [!" >a1~!!%;;
 Q!;!;
< !n >r*   c                   
 |syt        | j                        dkD  rt        j                  d       y| j                  d   }|j                  }|j
                  }t        j                  d|       t        |      t        |      t        |      z
  k(  r$t        j                  d       |t        |      d= yt        j                  d       i 
d}t        t        |            D ]  }||v r|dz  }||z
  
|<    t        j                  d
j                                
fd	} ||j                          ||j                         |D ]&  } ||j                          ||j                         ( | j                  r;| j                  d   }	t        |	j                  
       t        |	j                  
       t        |d
      D ]  }|j                  |        t        j                  d       y)zRemove tensors from model.Nr   zSkipping the removal of dangled tensors since the model has multiple subgraphs and tensors can be used in the different subgraph(s)r   z Removing tensors at indices : %sz3Removing tensors only at the end of the tensor listz,Removing tensors requires updating the modelzOld to new tensors map: %sc                 V    t        |       D ]  \  }}j                  |d      | |<    y )N)r   r#   )tensor_idxsr   tid_old_to_new_tensorss      r(   update_tensorsz2_remove_tensors_from_model.<locals>.update_tensorsi  s1    [) :%!R-11"b9A:r*   T)reversez#Removed tensors marked for deletion)r2   	subgraphsr   inforC   	operatorsdebugminr   __str__r`   rf   signatureDefsr   sortedpop)r   remove_tensors_idxssubgraphrC   r  left_shift_byr   r   rB   rg   r   s             @r(   _remove_tensors_from_modelr  J  s   	
ALL    __Q('  )	--24GH 		W4G0H!HHMMGH'()*MM@AMS\" 8	#	#$'-$7S!	8
 MM.0D0L0L0NO: 8??#8##$ !RYYRZZ ! ))!,m#M$8$8:NO#M$9$9;OP)48 kk#MM78r*   c                     |t         j                  k(  ry| j                  st        | dd|       yt	        | j                        D ]  \  }}t        | |j
                  ||        y)zModify model input type.Nr   r   )r   float32r  %_modify_model_input_type_per_subgraphr   subgraphIndex)r   inference_input_typesignature_indexrg   s       r(   _modify_model_input_typer  {  sf    V^^+
			)%B8LM
(1%2E2E(F Q$o})%1L1L*9;OQQr*   c                 4   | j                   |   }|j                  }|j                  }t        |       }|rJ|sH|j                  D ]8  }t        ||   j                        }	|	t        j                  k(  s/t        d       yg }
|D ]n  }|j                  |v s|j                  d   |j                  v s/||j                  d      ||j                  d      }}t        |j                        }|t        j                  k7  r4||k(  rt        dj                  |j                  t        |                  t        |j                        }|t        vrQt        dj                  |j                  t!        d t        j#                         D              t        |                  t        |   }||vr4t        dj                  t!        d |D              t        |                  |
j%                  |       q t'        |j                        t'        |
      k7  rt)        j*                  d	       |t        j,                  k(  r|
D ]  }||j                  d      j.                  }t1        j2                         }|j4                  d   g|_        |j6                  d   d
z   g|_        |||j                  d      _        t0        j8                  j:                  ||j                  d      _         y|t        v rt=               }|
D ]  }|j                  d   |j                  |j                  |j                  d   k(  <   |dk\  r}| j>                  |   }tA        t'        |j                              D ]M  }|j                  |   jB                  |j                  d   k(  s-|j                  d   |j                  |   _!        O |jE                  |j                  d          |jG                  |        tI        | |       yt        dj                  t        |                  )z%Modify model input type per subgraph.zModel input is not dequantized.Nr   zvInitial model input type must be tf.float32. Expected type for tensor with name '{}' is tf.float32, instead type is {}zqInitial model input is not quantized. Expected type for tensor with name '{}' should be in {}, instead type is {}c              3   2   K   | ]  }t        |        y wNr-   .0ts     r(   	<genexpr>z8_modify_model_input_type_per_subgraph.<locals>.<genexpr>        5a&q) 5   zOUnsupported `inference_input_type` value. Expected to be in {}, instead got {}.c              3   2   K   | ]  }t        |        y wr  r  r  s     r(   r  z8_modify_model_input_type_per_subgraph.<locals>.<genexpr>       H(+Hr  zFor model inputs containing unsupported operations which cannot be quantized, the `inference_input_type` attribute will default to the original type.   z,Unsupported `inference_input_type` value {}.)%r   rC   r  r   r`   r)   r<   r   r  r$   opcodeIndexrf   r%   r,   r-   _MAP_QUANT_TO_IO_TYPEStuplekeysr=   r2   r   warninguint8quantizationr   QuantizationParametersTscale	zeroPoint
TensorTypeUINT8r   r  r   r   r   remover  )r   subgraph_indexr  r  r	  rC   r  r   	input_idx
input_typeinput_quant_opsrB   float_tensorquant_tensor
float_type
quant_typeinference_io_typesint8_quantizationuint8_quantizationr  rg   r   s                         r(   r  r    s    __^,('  ) .e4(__ <	7	8J8O8OPj	v~~	%:;;<
  / !!b	~~**ryy|x/N#*299Q<#8'"**Q-:PLl78I8IJj	v~~	%--
HHN##%5j%AICD D 88I8IJj	1	1HHN!! 5,1135 5 ,	I./ 	/ 4J?'99$$*FH5GHH"#78%:; ;
 R C!!F 	S11OO	
 V\\) >!"**Q-0==$<<>"3"9"9!"<!=&7&A&A!&Ds&J%K"+=gbiil(#,#7#7#=#=gbiil > 55% 9;Ahoohoo156	A	++O<s=//01 	@A!!!$00BIIaL@24**Q-M  #/	@ biil+r u&9:
6==12	45 5r*   c                     |t         j                  k(  ry| j                  st        | dd|       yt	        | j                        D ]  \  }}t        | |j
                  ||        y)zModify model output type.Nr   r   )r   r  r  &_modify_model_output_type_per_subgraphr   r  )r   inference_output_typer  rg   s       r(   _modify_model_output_typer;    sf    fnn,
			*5!R9NO
(1%2E2E(F B$o}*5-2M2M+:+@BBr*   c                 
   | j                   |   }|j                  }|j                  }t        |       }|rJ|sH|j                  D ]8  }t        ||   j                        }	|	t        j                  k(  s/t        d       yg }
|D ]n  }|j                  |v s|j                  d   |j                  v s/||j                  d      ||j                  d      }}t        |j                        }|t        j                  k7  r4||k(  rt        dj                  |j                  t        |                  t        |j                        }|t        vrQt        dj                  |j                  t!        d t        j#                         D              t        |                  t        |   }||vr4t        dj                  t!        d |D              t        |                  |
j%                  |       q t'        |j                        t'        |
      k7  rt)        j*                  d	       |t        j,                  k(  rd
}t/        | j0                        D ];  \  }}t3        j4                  |      }|t6        j8                  j:                  k(  s9|} n |d
k(  rt7        j<                         }t6        j8                  j:                  |_        t6        j8                  j:                  |_         | j0                  j%                  |       t'        | j0                        dz
  }|
D ]  }||_
        ||j                  d      jB                  }t7        jD                         }|jF                  d   g|_#        |jH                  d   dz   g|_$        |||j                  d      _!        t6        jJ                  jL                  ||j                  d      _         y|t        v rtO               }|
D ]  }|j                  d   |j                  |j                  |j                  d   k(  <   |dk\  r}| jP                  |   }tS        t'        |j                              D ]M  }|j                  |   jT                  |j                  d   k(  s-|j                  d   |j                  |   _*        O |jW                  |j                  d          |jY                  |        t[        | |       yt        dj                  t        |                  )z&Modify model output type per subgraph.z Model output is not dequantized.Nr   zwInitial model output type must be tf.float32. Expected type for tensor with name '{}' is tf.float32, instead type is {}ztInitial model output is not dequantized. Expected type for tensor with name '{}' should be in {}, instead type is {}c              3   2   K   | ]  }t        |        y wr  r  r  s     r(   r  z9_modify_model_output_type_per_subgraph.<locals>.<genexpr>  r  r  zPUnsupported `inference_output_type` value. Expected to be in {}, instead got {}.c              3   2   K   | ]  }t        |        y wr  r  r  s     r(   r  z9_modify_model_output_type_per_subgraph.<locals>.<genexpr>(  r  r  zFor model outputs containing unsupported operations which cannot be quantized, the `inference_output_type` attribute will default to the original type.r   r   r  z-Unsupported `inference_output_type` value {}.).r   rC   r  r   rf   r)   r<   r   r  r$   r   r`   r%   r,   r-   r!  r"  r#  r=   r2   r   r$  r%  r   r   r   r   r   r   r   OperatorCodeTbuiltinCodedeprecatedBuiltinCoder&  r'  r(  r)  r*  r+  r   r  r   r   r   r,  r  )r   r-  r  r:  r	  rC   r  dequant_opcode_idxsoutputoutput_typeoutput_dequant_opsrB   r2  r1  r3  r4  r5  quant_opcode_idxr   r   r   quant_opr6  r7  r  rg   r   s                              r(   r9  r9    s    __^,('  ) 2%8*"" =89M9MNk		&;<<=
   "$b
--


1)))#*299Q<#8'"**Q-:PLl78I8IJj	v~~	%..
HHN##%5j%AICD D 88I8IJj	1	1HHN!! 5,1135 5 ,	I./ 	/ 4J? (::$$*FH5GHH"#89%;< <
 #E"$H 			c"455OO	
 fll* !4!45 V DDVLl	22;;	;	 2((*h&66??h'0'@'@'I'Ih$  *U001A5  ?'bn!"))A,/<<$<<>"3"9"9!"<!=&7&A&A!&Ds&J%K",>gbjjm)$-$8$8$>$>gbjjm!?  66%  <>IIaLhx''2::a=89	A	++O<s=0012 	@A""1%11RZZ]B3599Q<M!!!$0	@ bjjm,r u&9:
7>>23	56 6r*   c                     | j                   st        | dd       yt        | j                         D ]  \  }}t        | |j                  |        y)zAFinds back to back quantize ops and remove the first quantize op.r   r   N)r  +_remove_redundant_quantize_ops_per_subgraphr   r  )r   r  rg   s      r(   _remove_redundant_quantize_opsrJ  `  sR    			/q"=
(1%2E2E(F A$o}/0=0K0K0?AAr*   c                 $   | j                   |   }|j                  }|j                  }t        |       }t	        |       }g }i }	i }
|D ]  }|j
                  |v r|j                  |       ||j                  d      }||j                  d      }t        |j                        }t        |j                        }|t        j                  k7  r%|t        j                  k7  r||	|j                  d   <   |j
                  |v s|j                  d   |j                  v s||
|j                  d   <    |D ]  }|j                  d   }||	v s|	|   }| j                  rQ| j                  d   }|j                  D ]3  }|j                  |j                  d   k(  s |j                  d   |_        5 |j                  d   }|j                  d   |j                  d<   |D ]V  }||j                  v s||k7  st        |j                        D ]'  \  }}||k(  s|j                  d   |j                  |<   ) X |j!                  |        |D ]  }|j                  d   }||
v s|
|   }|j                  d   |j                  |j                  |j                  d   k(  <   |dk\  rQ| j                  |   }|j                  D ]3  }|j                  |j                  d   k(  s |j                  d   |_        5 |j!                  |       |j!                  |        y)z+Remove redundant quantize ops per subgraph.r   N)r   rC   r  r   r   r   r=   r`   rf   r)   r<   r   r  r  r   r   r,  )r   r-  r  r	  rC   r  r   rB  all_quant_opsredundant_quant_tensorsoutput_dequant_tensorsrB   input_tensoroutput_tensorr/  rD  output_tensor_idxrequantize_oprg   rC  deleted_tensorop_userr   
dequant_ops                           r(   rI  rI  l  s    __^,('  ) .e41%8 - 0b	~~**2RYYq\*lbjjm,m78I8IJj89K9KLk	v~~	%+*G02		!-
--


1)))-/RYYq\*0  b

133-.?@m			++A.#++ 	.F2::a=0!#1F	. %++A.n "		!m1 ='W^^+=0H#,W^^#< =c<~-$1$9$9!$<gnnS!==
 r%*  #b

122)*;<jDFIIaLhx'':+=+=a+@@A	A	++O<#++ 	.F:#5#5a#88!#1F	. rz"#r*   c                     |t         j                  k(  r|t         j                  k(  r| S t        |       }t        ||       t	        ||       t        |       t        |      S )a  Modify the input/output type of a tflite model.

  Args:
    model: A tflite model.
    inference_input_type: tf.DType representing modified input type.
      (default tf.float32. If model input is int8 quantized, it must be in
      {tf.float32, tf.int8,tf.uint8}, else if model input is int16 quantized,
      it must be in {tf.float32, tf.int16}, else it must be tf.float32)
    inference_output_type: tf.DType representing modified output type.
      (default tf.float32. If model output is int8 dequantized, it must be in
      {tf.float32, tf.int8,tf.uint8}, else if model output is int16 dequantized,
      it must be in {tf.float32, tf.int16}, else it must be tf.float32)
  Returns:
    A tflite model with modified input/output type.

  Raises:
    ValueError: If `inference_input_type`/`inference_output_type` is unsupported
      or a supported integer type is specified for a model whose input/output is
      not quantized/dequantized.
    RuntimeError: If the modification was unsuccessful.

  )r   r  r   r  r;  rJ  r   )r   r  r:  r   s       r(   modify_model_io_typerW    sT    2 fnn,v~~-L8?,<)=>L*?@ .	0	>>r*   c                    | r| j                   sg S t               }| j                  D ]  }|j                  D ]  }|j                  s|j                  j
                  j                  dk(  s|j                  j
                  s*|j                  t        j                  j                         s|j                  t        j                  j                           t        |      S )zGet sparsity modes used in a tflite model.

  The sparsity modes are listed in conversion_metadata.fbs file.

  Args:
    model_object: A tflite model in object form.

  Returns:
    The list of sparsity modes used in the model.
  r   )metadatar   r   rC   sparsityblockMapsizer   conversion_metadata_fbModelOptimizationModeRANDOM_SPARSITYBLOCK_SPARSITYlist)r   resultr	  r3   s       r(   get_sparsity_modesrc    s     
\22I5&(( Ih"" I__ 
	!	!	&	&!	+6??3K3K

"88HH	J 	

"88GG	III 
fr*   c                 @   d}| j                   D ]  }|j                  ut        |t        |j                              }|j                  D ]G  }|j                  t        ||j                        }|j                  2t        ||j                        }I |j                  t        |t        |j                              }|j                  D ]u  }|j                  D| j                  |j                     }|j                  t        |t        |j                              }|j                  `t        ||j                        }w |j                  t        |t        |j                              }|j                  mt        |t        |j                              } |S )zCalculate a 64-bit integer hash for a TensorFlow Lite model based on its structure.

  Args:
      model: A TensorFlow Lite model object.

  Returns:
      int: A 64-bit integer hash value representing the model structure.
  r   )r   r   update_hash_with_primitive_valuer2   r`   update_hash_with_arrayrf   rC   bufferbuffersr   rI   )r   
hash_valuer	  operatorr3   rg  s         r(   get_model_hashrk    s    *// %h%3
c(,,-j (( L(??&-j(//J*'-j(:J:JK*L #3
c(**+j $$ 	H&==$==/&[[$9C,J <<#-j&,,G*	H "3
c(//*j #3
c(**+jG%N 
r*   c                 p   t        j                  d      }t        j                  |       } t        j                  |      }t        j                  | g      } t        j                  |g      }t        j                  | ||z   t        j                  | d      z   t        j
                  | d      z         d   } | S )zUpdate the hash value using a primitive value.

  Args:
      hash_value (uint64): The current hash value.
      value: The primitive value to incorporate into the hash.

  Returns:
      int: The updated hash value.
  l    x~fq	 r   r   r   )npuint64rq   bitwise_xor
left_shiftright_shift)ri  rj   
hash_consts      r(   re  re  ,  s     yy+,*yy$*
))E
% xx%*
((E7
% ~~
MM*b)* NN:q)* * 
r*   c                 0    ||D ]  }t        | |      }  | S )zUpdate the hash value using a TFLite int array.

  Args:
      hash_value (int): The current hash value.
      int_array: A TFLite int array to incorporate into the hash.

  Returns:
      int: The updated hash value.
  )re  )ri  	int_arrayr   s      r(   rf  rf  L  s-      C3JBjC	r*   c                    	 t        j                  d      }|j                  |j                  |             t	        j
                         }|j                         |_        | j                  sg | _        nP| j                  D ]A  }|j                  j                  d      t        k(  s&|| j                  |j                  <   | c S  | j                  sg | _        | j                  j                  |       t	        j                         }t        |_	        t!        | j                        dz
  |_        | j                  j                  |       | S # t"        $ r | cY S w xY w)zAdd or update conversion metadata to a tflite model.

  Args:
    model_object: A tflite model in object form.
    metadata: The conversion metadata.

  Returns:
    A tflite model object with embedded conversion metadata.
  r   utf-8r   )r   r   r   r   r   BufferTr   r   rY  r,   decodeCONVERSION_METADATA_FIELD_NAMErh  rg  r=   	MetadataTr2   	Exception)r   rY  metadata_builderbuffer_fieldmetametadata_fields         r(   populate_conversion_metadatar  \  s-   "**1-HMM*:;<$$&L(//1L   l '' $99G$(FF.:,

t{{
+

 l-((*N8N 4 459N  0	 s   B#E &E BE EEc                    t        j                  |       }|r|j                  sy|j                  D ]  }|j                  j	                  d      t
        k(  s&|j                  |j                     j                  j                         }t        j                  j                  t        j                  j                  |d            c S  y)zRead conversion metadata from a tflite model.

  Args:
    model_buffer: A tflite model.

  Returns:
    The conversion metadata or None if it is not populated.
  Nrv  r   )r   convert_bytearray_to_objectrY  r,   rx  ry  rh  rg  r   tobytesr]  ConversionMetadataTr   ConversionMetadataGetRootAsConversionMetadata)model_bufferr   r~  metadata_bufs       r(   get_conversion_metadatar    s     "==lK,	\22## dyy $BB!))$++6;;CCEl#77CC
 
3
3
O
OA  
r*   r  )P   NNF)a__doc__r   r   r   abslr   r   numpyrm  tensorflow.core.protobufr   rP   r   r^   tensorflow.lite.pythonr   r]  r   r   r   r	   _tflite_keras_utiltensorflow.lite.python.op_hintr
   r   tensorflow.lite.toolsr   tensorflow.python.eagerr   tensorflow.python.frameworkr   rv   r   r   r   tensorflow.python.grapplerr    tensorflow.python.training.saverr   r]   ry  model_input_signaturetrace_model_callget_save_specjaxr   _jitImportErrorr  float16int32r%  int64stringboolint16	complex64int8float64
complex128uint32r"   r   r!  r)   r-   r5   rE   rN   rX   rs   r{   r   ru   r   r   r   r   r   r   r   r   r   r  r  r  r;  r9  rJ  rI  rW  rc  rk  re  rf  r  r  r   r*   r(   <module>r     s&   2   
    > F d C . J D G 2 , U . S 3 T "7  +@@ %66 "00
 ~~~~||||||}}{{||{{    "  KK&++v||,
LL6<<. (3
*(V$>( #'	'8T&R$$N&2B .0,0+/5:u"p!>.9b :@ QY5x <B>> Bi6X	A>#D !' ..%?P@4n@ $N["  	$s   H HH