
    hF              
          d dl Z d dlZd dlmZmZ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  e       rd dlmZ ddlmZ dd	lmZmZmZ erdd
lmZ ddlmZ ddlmZ  G d d      Z G d de      Z G d de      Z G d de
j>                        Z  G d de
j>                        Z! G d d      Z" G d d      Z# G d de      Z$ G d de      Z% G d d e      Z&d!e'e(ef   d"e)d#e*d$e'e(ef   fd%Z+d!e'e(ef   d"e)d$e'e(ef   fd&Z,y)'    N)TYPE_CHECKINGAnyOptional   )prune_linear_layer)is_sklearn_available)	roc_curve)isin_mps_friendly   )LogitsProcessorListMinLengthLogitsProcessorSuppressTokensLogitsProcessor)PreTrainedModel)PreTrainedTokenizerBase)GenerationConfigc                       e Zd ZdZdej
                  deej
                  eej                     f   fdZ	dej
                  dej                  de
fdZy)	CandidateGeneratorz`Abstract base class for all candidate generators that can be applied during assisted generation.	input_idsreturnc                 2    t        | j                   d      )ag  
        Fetches the candidates to be tried for the current input.

        Args:
            input_ids (`torch.LongTensor` of shape `(batch_size, sequence_length)`):
                Indices of input sequence tokens in the vocabulary. [What are input IDs?](../glossary#input-ids)

        Return:
            `torch.LongTensor` of shape `(batch_size, candidate_length)` containing the candidate sequences to be
            assessed by the model and, optionally, a `torch.FloatTensor` of shape `(batch_size, candidate_length,
            vocabulary_size)` containing the logits associated to each candidate.
        zT is an abstract class. Only classes inheriting this class can call `get_candidates`.NotImplementedError	__class__)selfr   s     i/var/www/html/eduruby.in/venv/lib/python3.12/site-packages/transformers/generation/candidate_generator.pyget_candidatesz!CandidateGenerator.get_candidates,   s!     "~~rs
 	
    scoresnum_matchesc                 2    t        | j                   d      )  
        Updates the candidate generation strategy based on the outcomes.

        Args:
            input_ids (`torch.LongTensor` of shape `(batch_size, sequence_length)`):
                Indices of input sequence tokens in the vocabulary. [What are input IDs?](../glossary#input-ids)
            scores (`torch.FloatTensor` of shape `(batch_size, candidate_length, config.vocab_size)`):
                Prediction scores of a language modeling head. These can be logits for each vocabulary when not using
                beam search or log softmax for each vocabulary token when using beam search
            num_matches (`int`):
                The number of matches between the candidate sequences and the model predictions.
        z_ is an abstract class. Only classes inheriting this class can call `update_candidate_strategy`.r   r   r   r   r   s       r   update_candidate_strategyz,CandidateGenerator.update_candidate_strategy=   s%     "~~ + +
 	
r   N)__name__
__module____qualname____doc__torch
LongTensortupler   FloatTensorr   intr#    r   r   r   r   )   sb    j
(8(8 
U5CSCSU]^c^o^oUpCp=q 
"
53C3C 
UM^M^ 
mp 
r   r   c                      e Zd ZdZ	 	 ddej
                  ddddded	eej                     d
dfdZ	dej
                  de
ej
                  eej                     f   fdZdej
                  dej                  defdZdej
                  de
eef   fdZ	 ddej
                  dededefdZdej
                  dededefdZdede
ej
                  eej                     f   fdZy)AssistedCandidateGeneratorar  
    `CandidateGenerator` class to be used for assisted generation and speculative decoding. This class generates
    candidates through the use of a smaller model. Read the following blog post for more information:
    https://huggingface.co/blog/assisted-generation

    Args:
        input_ids (`torch.LongTensor` of shape `(batch_size, sequence_length)`):
            Indices of input sequence tokens in the vocabulary. [What are input IDs?](../glossary#input-ids)
        assistant_model (`PreTrainedModel`):
            The model to be used for generating candidates. This model should be smaller than the main model.
        generation_config (`~generation.GenerationConfig`, *optional*):
            The generation configuration to be used as base parametrization for the generation call.
        logits_processor (`LogitsProcessorList`):
            An instance of [`LogitsProcessorList`]. List of instances of class derived from [`LogitsProcessor`]
            used to modify the prediction scores of the language modeling head applied at each generation step.
        model_kwargs (`Dict`):
            The keyword arguments that will be passed to the main model, and are used as base inputs for the assistant
            model as well.
        inputs_tensor (`torch.Tensor`, *optional*):
            The model input tensor. In encoder-decoder models, this is the encoder input.
    Nr   assistant_modelr   generation_configr   model_kwargsinputs_tensorlogits_processorr   c                    |j                   }|j                  |      }||j                  |      }|| _        |j                  j                  | _        |j                  j
                  | _        |j                  | j                  j                  _        i }|j                         D ][  \  }	}
|	dvst        |
t        j                        r|
j                         j                  |      nt        j                  |
      ||	<   ] d|v r|j                         s|d= |j                  j                   rJ|j#                  ||j                  j$                  |      \  }}}|j'                  ||||j                        }nd|v r|d   |d<   || _        |j                  j                   rd| _        nyd|v rnd| _        | j(                  j-                  dt        j.                  |j0                  d   df|j                   t        j2                  	            | j(                  d
<   nd| _        ||n	t5               | _        t        j                  |      | _        d| j                  _        d| j                  _        | j
                  | j                  _        d| j                  _        | j                  j>                  | _         d| j                  _        d | j                  _!        | j6                  D ]  }t        |tD              stG        d       d | j                  _$        tK               rB| j                  j                  j
                  r!tM        |       tN        u rg | _(        g | _)        y y y y )N)encoder_outputspast_key_valueslogits_to_keepr6   decoder_input_idsr   decoder_attention_maskr   r   )devicedtypeattention_maskTzPassing `MinLengthLogitsProcessor` when using `assisted_generation is disabled. Please pass in `min_length` into `.generate()` instead)*r;   tor0   r1   num_assistant_tokensassistant_confidence_thresholdeos_token_iditems
isinstancer(   Tensordetachcopydeepcopy_supports_logits_to_keepconfigis_encoder_decoder_prepare_model_inputsbos_token_id._prepare_encoder_decoder_kwargs_for_generationassistant_kwargsinput_ids_keygetonesshapelongr   r4   return_dict_in_generateoutput_scoresis_assistant
min_lengthmain_model_min_lengthmin_new_tokensr   
ValueErrorcache_implementationr   typer/   probsmatches)r   r   r0   r1   r2   r3   r4   r;   rN   keyvaluemodel_input_name	processors                r   __init__z#AssistedCandidateGenerator.__init__g   s7    !''LL(	$),,V4M  /$3$E$E$Z$Z!.=.O.O.n.n+ ?P>\>\..; &,,. 	JC@@1;E5<<1PELLN%%f-VZVcVcdiVj !%	 //8`8`8b !12 !!44@O@e@e@@MMO_A=M+-=  /]]/1A?CdCd  ,.2>?P2Q./ 0 !!44!4D"22!,D6:6K6K6O6O(

IOOA.29;K;KSXS]S]^7D!!"23 "-D 5E4P 0ViVk!%/@!A9=6/3,@D@c@c=.2+ &*%;%;%F%F",-)04-.. 	I)%=> M 	 7;3 !"$$66UUT
88DJDL 9 V #r   r   c                     |j                  | j                  j                        }| j                  |      \  }}|dk(  r|dfS | j	                  |       | j                  |||      }| j                  |      \  }}||fS )Z  
        Fetches the candidates to be tried for the current input.

        Args:
            input_ids (`torch.LongTensor` of shape `(batch_size, sequence_length)`):
                Indices of input sequence tokens in the vocabulary. [What are input IDs?](../glossary#input-ids)

        Return:
            `torch.LongTensor` of shape `(batch_size, candidate_length)` containing the candidate sequences to be
            assessed by the model and a `torch.FloatTensor` of shape `(batch_size, candidate_length,
            vocabulary_size)` containing the logits associated to each candidate.
        r   N)r>   r0   r;   _calculate_new_tokens_update_past_and_masks_prepare_generation_args_generate_candidates)r   r   rY   max_new_tokensgeneration_argscandidate_idscandidate_logitss          r   r   z)AssistedCandidateGenerator.get_candidates   s     LL!5!5!<!<=	)-)C)CI)N&Qd?"##I.77	>Sab*.*C*CO*T''...r   r   r   c                    | j                   j                  j                  dv rH|t        |d         dz
  k(  r| xj                  dz  c_        nt        d| j                  dz
        | _        t               rx| j                   j                  j                  rVt        |       t        u rC| j                  j                  dg|z         t        | j                        t        | j                        kD  r| j                  j                  d       t        | j                        t        | j                        z
  }|dkD  r| j                  | d= t        | j                        dkD  rddhj                  | j                        rgt        | j                  | j                        \  }}}d|z
  }|d|z  z   }	t!        j"                  |	      }
||
   }|| j                   j                  _        yyyyyy)	r!   >   	heuristicheuristic_transientr   r   g       @g      ?N      )r0   r1   num_assistant_tokens_schedulelenr?   maxr   r@   r\   r/   r^   extendr]   appendissubsetr	   npargmin)r   r   r   r   excess_lengthfprtpr
thresholdsfnrcostsoptimal_threshold_indexbest_thresholds               r   r#   z4AssistedCandidateGenerator.update_candidate_strategy   s     11OO T
 

 c&)nq00))S0),/T5N5NQT5T,U)
 !"$$66UUT
88 LLk 124::T\\!22##A&  

Oc$,,.??Mq JJ~/ DJJ!#A(E'0tzz'J$S*#g a#g +-))E*:'!+,C!DXf$$66U )F# 9 V #r   c                     |j                   d   }t        t        | j                        | j                  j
                  |z
  dz
        }t        t        || j                  |z
        d      }||fS )zCCalculate the minimum and maximum number of new tokens to generate.r   r   )rR   minr,   r?   r1   
max_lengthru   rX   )r   r   new_cur_lenrj   rY   s        r   rf   z0AssistedCandidateGenerator._calculate_new_tokens  si    oob)S!:!:;T=S=S=^=^al=lop=pqS1K1Kk1YZ\]^~--r   remove_from_pkvnum_added_tokensc                    | j                   j                  dd      du}|r|j                  d   dz
  |z
  }| j                   d   j                  ||z
         t	        | j                   |j                  d   | j
                  j                  j                        | _         t        | j                   |j                  d         | _         |S )zLUpdate past key values and attention masks for subsequent generation rounds.r7   Nr   r   )	rN   rP   rR   crop_prepare_attention_maskr0   rI   rJ   _prepare_token_type_ids)r   r   r   r   has_past_key_valuesnew_cache_sizes         r   rg   z1AssistedCandidateGenerator._update_past_and_masks   s     #33778I4PX\\&__R014FN!!"3499.K[:[\$;%%yr':D<P<P<W<W<j<j%D! %<D<Q<QS\SbSbceSf$gD!""r   rY   rj   c           
      V    | j                   |d|d|d| j                  d| j                  iS )z*Prepare arguments for the generation call.rY   rj   r1   r4   )rO   r1   r4   )r   r   rY   rj   s       r   rh   z3AssistedCandidateGenerator._prepare_generation_args/  s9     	nn!7!7 5 5
 	
r   rk   c                     | j                   j                  di || j                  }|j                  | j                  d<   t	               r| j                   j
                  j                  rt        |       t        u rt        j                  |j                  d      }t        j                  |d      }|j                  dt        |j                         df   }|t        t        |            |f   }| j                   j#                  |j%                                t        j&                  |j                  d      }|j                  }||fS )z7Generate candidate sequences using the assistant model.r7   r   dimr   Nr   r-   )r0   generaterN   r7   r   r1   r@   r\   r/   r(   catr   softmax	sequencesrt   ranger]   rv   toliststack)	r   rk   assistant_outputscores_tensorscores_softmaxidsprm   rl   s	            r   ri   z/AssistedCandidateGenerator._generate_candidates9  s   84//88d?ddNcNcd3C3S3S/0 "$$66UUT
88!II&6&=&=1EM"]]=bAN",,R#6F6M6M2N1N1P-PQCuSX34AJJahhj) ;;'7'>'>AF(22...r   NN)r   r   )r$   r%   r&   r'   r(   r)   dictr   rD   rc   r*   r+   r   r,   r#   rf   boolrg   rh   ri   r-   r   r   r/   r/   P   st   8 1526]##] +] .	]
 ]  -] 0]~/(8(8 /U5CSCSU]^c^o^oUpCp=q /28g53C3C 8gUM^M^ 8gmp 8gt.u/?/? .E#s(O . ^_#))#<?#WZ#	#
%2B2B 
TW 
il 
qu 
/D /U5CSCSU]^c^o^oUpCp=q /r   r/   c                       e Zd ZdZ	 	 ddej
                  dddddddd	d
edeej                     ddf fdZ	e
d        Ze
d        Ze
d        Zd Zdej
                  deej
                  eej                      f   fdZdej
                  deej
                  ef   fdZdej
                  dej
                  dej
                  dej
                  fdZ xZS )-AssistedCandidateGeneratorDifferentTokenizersa  
    `CandidateGenerator` class to be used for Universal Assisted Generation (UAD): assisted generation with different tokenizers
    for the assistant and main models. This class generates candidates through the use of a smaller
    model.

    The main model input tokens are re-encoded into assistant model tokens, then candidate tokens are generated in the assistant encoding, which are
    in turn re-encoded into main model candidate tokens. Validation then proceeds as explained above.
    The re-encoding steps involve decoding token ids into text and then encoding the text using a different tokenizer.
    Since re-encoding the tokens may result in tokenization discrepancies, UAD finds the longest common subsequence between the source and target encodings,
    to ensure the new tokens include the correct prompt suffix.

    Args:
        input_ids (`torch.LongTensor` of shape `(batch_size, sequence_length)`):
            Indices of input sequence tokens in the vocabulary. [What are input IDs?](../glossary#input-ids)
        assistant_model (`PreTrainedModel`):
            The model to be used for generating candidates. This model should be smaller than the main model.
        target_tokenizer (`PreTrainedTokenizerBase`):
            The tokenizer used for the target model.
        assistant_tokenizer (`PreTrainedTokenizerBase`):
            The tokenizer used for the assistant model.
        generation_config (`~generation.GenerationConfig`, *optional*):
            The generation configuration to be used as base parametrization for the generation call.
        logits_processor (`LogitsProcessorList`):
            An instance of [`LogitsProcessorList`]. List of instances of class derived from [`LogitsProcessor`]
            used to modify the prediction scores of the language modeling head applied at each generation step.
        model_kwargs (`Dict`):
            The keyword arguments that will be passed to the main model, and are used as base inputs for the assistant
            model as well.
        inputs_tensor (`torch.Tensor`, *optional*):
            The model input tensor. In encoder-decoder models, this is the encoder input.
    r   r0   r   target_tokenizerr   assistant_tokenizerr1   r   r2   r3   r4   r   c	                     t         	|   ||||||       || _        || _        d | _        d | _        |j                  j                  | _        |j                  j                  | _        y N)	superrc   r   r   prev_target_ids_lenprev_assistant_idsr1   target_lookbehindassistant_lookbehind)
r   r   r0   r   r   r1   r2   r3   r4   r   s
            r   rc   z6AssistedCandidateGeneratorDifferentTokenizers.__init__m  sg     	O5FVceuv 0#6 26 "&!0!B!B!T!T$3$E$E$Z$Z!r   c                    t               }i }|D ]  }t        j                  |      }t        |j	                               }||v r6|j                  |       d}|dz  }|d   | j                  d   k  r|d   | j                  d   k  rqt        |j	                               }|j                  |       | |d   |d   f   dk(  r|dz  }|dz  }nn+|d   | j                  d   k  r|d   | j                  d   k  rq|||<    |S )a  
        Calculates the length of the longest diagonal sequence in a given matrix.
        Args:
            input_matrix (torch.Tensor): The input matrix.
            nonzero_idx (torch.Tensor): The indices of the non-zero elements in the matrix.
        Returns:
            dict: A dictionary where the keys are the indices of the non-zero elements and the values are the lengths of the longest diagonal sequences starting from those indices.
        r   r   )setr(   cloner*   r   addrR   )input_matrixnonzero_idxvisiteddiagsidx	start_idxtuple_start_idxcur_diag_lens           r   _get_longest_diag_dictzDAssistedCandidateGeneratorDifferentTokenizers._get_longest_diag_dict  s$    % 	&CC(I#I$4$4$67O')KK(LNIA,!3!3A!669Q<,J\J\]^J_;_"'	(8(8(:";O,	!il :;q@ A%LNI A,!3!3A!669Q<,J\J\]^J_;_ &E#J)	&* r   c                     t         j                  | | j                               }t        |j	                               }t        |j                               }t        j                  |      }||   }||   }||fS )a  
        Returns the start index and length of the longest diagonal in the given input.
        Args:
            input_matrix (numpy.ndarray): The input matrix.
        Returns:
            tuple: A tuple containing the start index and length of the longest diagonal.
        )r   r   nonzerolistvalueskeysry   argmax)r   r   diags_values
diags_keys	best_diagdiag_start_indexdiag_start_lengths          r   _get_longest_diag_indexzEAssistedCandidateGeneratorDifferentTokenizers._get_longest_diag_index  st     >TT,..0
 ELLN+%**,'
IIl+	%i0(3!222r   c                    |j                   | k(  }t        j                  |      st        j                  |      }|j	                  t
              }|j                         j                         syt        j                  |      \  }}|d   |z   }|d   |z   }| j                  d   |z
  j                         }|dd||z   df   }	|dd|||z   f   }
||	|
fS )a  
        Input:
            prompt: 2D array of shape (batch_size, prompt_length), represents the original prompt tokens
            prompt_plus_new_tokens: 2D array of shape (batch_size, prompt_length), represents the suffix of the original prompt, with additional new tokens.
        Output:
            discrepancy_length: int, represents the number of tokens that need to be replaced from prompt
            new_tokens_only: 2D array of shape (batch_size, new_token_length), represents the new tokens that are not in prompt
            discrepancy_only: 2D array of shape (batch_size, discrepancy_length), represents the new tokens that are in prompt but not in prompt_plus_new_tokens
        )NNNr   r   N)Tr(   	is_tensortensorr>   r,   anyitemr   r   rR   )promptprompt_plus_new_tokenscompare_matcompare_mat_intlongest_locationlongest_diag_lengthnew_token_start_indexdiscrepancy_with_olddiscrepancy_lengthnew_tokens_onlydiscrepancy_onlys              r   _get_tokens_diagz>AssistedCandidateGeneratorDifferentTokenizers._get_tokens_diag  s     -..&8{+,,{3K%..-""$))+#0]0u0u1
-- !1 36I I/25HH$ll1o0DDJJL04IL^4^4`1`a1$'<?Q'QQQ
 "?4DDDr   c                 |    |j                  |dd      } ||dd      d   }|j                  |j                        S )a$  
        Convert token IDs from one tokenizer to another.
        Args:
            input_ids: The input token IDs.
            source_tokenizer: The source tokenizer.
            destination_tokenizer: The destination tokenizer.
        Returns:
            The converted token IDs.
        Tskip_special_tokensclean_up_tokenization_spacesptadd_special_tokensreturn_tensorsr   )batch_decoder>   r;   )r   r   source_tokenizerdestination_tokenizertextdest_idss         r   &convert_source_tokens_to_target_tokenszTAssistedCandidateGeneratorDifferentTokenizers.convert_source_tokens_to_target_tokens  sF      ,,YDos,t($W[\]hi{{9++,,r   r   c                    t        | j                        }|dk(  r|dfS |j                  | j                  j                        }d}| j                  |      \  }}|| _        t        t        || j                  |j                  d   z
        d      }| j                  ||       | j                  |||      }| j                  j                  dd        | j                  j                  di || j                  }| j!                  ||j"                  |      }|j                  d   | _        |j&                  | j                  d<   |j"                  | _        | j$                  |j                  d   k\  r|dfS |dfS )re   r   Nr   r=   r   r7   r-   )r,   r?   r>   r0   r;   _prepare_assistant_input_idsr   ru   r   rX   rR   rg   rh   rN   popr   _process_assistant_outputsr   r   r7   )	r   r   rj   r   assistant_input_idsrY   rk   r   new_target_idss	            r   r   z<AssistedCandidateGeneratorDifferentTokenizers.get_candidates  sw    T667Qd?"LL!5!5!<!<=	/3/P/PQZ/[,_"5S1K1KNaNgNghjNk1klnop##$7I778K^]kl!!"2D984//88d?ddNcNcd88DTD^D^`st $-??1#5 3C3S3S/0"2"<"<##~';';A'>>d?"t##r   c                 r   | j                   | j                  d}d}| j                  e| j                  | j                  kD  rK| j                  | j                  z
  } | j
                  |dd|df   fi |}|j                  d   }| j                  dd| df   }| j                  ||      \  }}	}
| j                  }|	|dkD  r||
j                  d   dkD  rj||
j                  d   k(  r|
|dd| df<   nI||
j                  d   kD  r7||
j                  d   z
  }|ddd| f   }|
|dd|
j                  d    df<   |}|	j                  d   dkD  r6t        j                  ||	gd      }||fS t        j                  ||gd      }||fS  | j
                  |fi |}|j                  d   | _        ||fS )zIConverts target input IDs to assistant input IDs, handling discrepancies.r   r   r   Nr   r   r   )
r   r   r   r   r   r   rR   r   r(   r   )r   r   convert_kwargsr   start_index_in_target_windownew_assistant_idsprompt_use_length
prompt_user   r   r   r   discrepancy_length_diffs                r   r   zJAssistedCandidateGeneratorDifferentTokenizers._prepare_assistant_input_ids  s4    !% 5 5%)%=%=
 "".43K3KdNdNd3d+/+C+CdF\F\+\( K K K!9::;!?M! !2 7 7 :005F4F4G1GHJDHDYDY-EA1A #'"9"9*%).>.D.DQ.G!.K)-=-C-CA-FFGW+A0B/B/C,CD+.>.D.DQ.GG2DGWG]G]^_G`2`/.A!E^G^F^E^B^._+O_+A0@0F0Fq0I/I/K,KL&8O"((+a/*/))5H/4Z`b*c' #O33 ',ii1DFW0X^`&a#
 #O33 #N$"M"Mi"j[i"j'0q'9D$"O33r   assistant_sequencesr   c                    | j                   j                  d   }|| j                  z
  }| j                  |dd|df   | j                  | j
                        }|j                  d   }|dd| df   }| j                  ||      \  }	}
}	|}|
,|
j                  d   dkD  r3t        j                  ||
gd      }nt        j                  ||gd      }t        | j                  d      r|ddd| j                  j                  f   }|S )z7Processes assistant outputs to obtain target input IDs.r   Nr   r   r   r   r   )r   rR   r   r   r   r   r   r(   r   hasattrr1   r   )r   r   r   r   num_prev_assistantstart_assistant_look_indexnew_target_ids_from_windowtarget_prompt_use_lengthtarget_prompt_use_target_new_tokens_onlyr   s               r   r   zHAssistedCandidateGeneratorDifferentTokenizers._process_assistant_outputsE  s1    "44::1=%7$:S:S%S"%)%P%P#=#> >?!55"&"7"7 &Q &
"
 $>#C#CA#F %a*B)B)C&CD'+'<'<=NPj'k$!1"!-%++A.2!&N<R+SY[!\ #YY8R'SY[\N4))<8+A/R1G1G1R1R/R,RSNr   r   )r$   r%   r&   r'   r(   r)   r   r   rD   rc   staticmethodr   r   r   r   r*   r+   r   r,   r   r   __classcell__r   s   @r   r   r   L  sa   P 1526[##[ +[ 4	[
 7[ .[ [  -[ 0[( ! !F 3 3& E E@-&($(8(8 ($U5CSCSU]^c^o^oUpCp=q ($T,4e6F6F ,45QVQaQacfQfKg ,4\))@E@P@Pglgwgw			r   r   c                   (     e Zd ZdZ fdZd Z xZS )_PruneReindexingLMHeadal  
    A class to prune and reindex the language model head.

    This class prunes the language model head to only include the specified token IDs and reindexes the logits
    to map back to the original vocabulary.

    Args:
        original_lm_head (nn.Module): The original language model head.
        token_ids (list[int]): The list of token IDs to keep.
    c                     t         |           t        ||      j                  |j                  j
                        | _        y r   )r   rc   r   r>   weightr<   pruned_lm_head)r   original_lm_headassistant_overlap_token_idsr   s      r   rc   z_PruneReindexingLMHead.__init__r  s9    01AC^_bb##))
r   c                 (    | j                  |      }|S r   )r  )r   hidden_statespruned_logitss      r   forwardz_PruneReindexingLMHead.forwardx  s    ++M:r   )r$   r%   r&   r'   rc   r  r   r  s   @r   r  r  f  s    	
r   r  c                   p     e Zd Zdej                  f fdZdej                  dej                  fdZ	 xZ
S )_MapInputEmbeddingoriginal_embeddingc                 n    t         |           || _        |j                  | _        || _        d| _        y)aT  
        Wraps an existing embedding layer and remaps token IDs before lookup.

        Args:
            original_embedding (nn.Embedding): Pre-trained or existing embedding layer.
            assistant_overlap_token_ids (dict): Mapping from original token IDs to new token IDs.
                          Example: {old_id: new_id}
        FN)r   rc   r  r  r  map)r   r  r  r   s      r   rc   z_MapInputEmbedding.__init__~  s4     	"4(//+F(r   r   r   c                     | j                   r1| j                  |d      j                  d      j                  d      }n	d| _         |}| j                  |      S )z
        Args:
            input_ids (torch.LongTensor): Tensor of token IDs (batch_size, seq_len).

        Returns:
            torch.FloatTensor: Corresponding input embeddings.
        )r   r   r   T)r  r  	unsqueezer  )r   r   my_input_idss      r   r  z_MapInputEmbedding.forward  sU     88;;Ie<LMWWXYZddefgLDH$L&&|44r   )r$   r%   r&   nn	Embeddingrc   r(   r)   r+   r  r   r  s   @r   r  r  }  s2    2<< 5!1!1 5e6G6G 5r   r  c                       e Zd ZU dZ ed       Zeed<   dZeed<   	 	 dddd	dd
ede	d   de
f
dZd Zd Zdee   fdZdej"                  dej"                  fdZdej&                  dej&                  fdZy)AssistantToTargetTranslatora  
    Translates token ids and logits between assistant and target model vocabularies. This class is used to handle
    vocabulary mismatches when using different tokenizers for the assistant and target models in speculative decoding,
    as introduced in the paper "Lossless Speculative Decoding Algorithms for Heterogeneous Vocabularies"
    (https://huggingface.co/papers/2502.05202).
    It maintains mappings between the two vocabularies and handles token/logit conversion.

    Args:
        target_tokenizer (`PreTrainedTokenizerBase`):
            The tokenizer used by the target (main) model.
        assistant_tokenizer (`PreTrainedTokenizerBase`):
            The tokenizer used by the assistant model.
        target_vocab_size (`int`):
            The size of the target model's vocabulary. If not provided, will be inferred from the target tokenizer.
        assistant_model (Optional[PreTrainedModel], optional): The assistant model to be used. Defaults to None for backward compatibility.
        assistant_prune_lm_head (bool): Whether to prune the assistant model's language model
            head to match the target vocabulary. This is only applicable if `assistant_model` is provided.
            Defaults to False for backward compatibility.
    InfFILTER_VALUEr   SUPPRESS_TOKEN_IDNr   r   r   target_vocab_sizer0   r   assistant_prune_lm_headc                 2   || _         || _        ||j                  nd| _        || _        | j                         \  | _        | _        | j                         | _	        d | _
        |xr |d u| _        t        | j                        dkD  r| j                  rt        j                  t        | j                  j!                               t        j"                  | j                        | _        |j'                         }t)        || j$                        }~|j+                  |       |j-                         }t/        || j$                        }	~|j1                  |	       |	| _        y t5        t7        | j                         | j                        g      | _
        y y )Ncpur   )r<   r;   )_target_tokenizer_assistant_tokenizerr;   _assistant_model_devicer  "_get_assistant_to_target_input_ids_assistant_to_target_input_idstarget_to_assistant_input_ids_get_suppress_input_ids_suppress_input_idslogits_processorsr  rt   r(   r   r   r   rS   r  get_output_embeddingsr  set_output_embeddingsget_input_embeddingsr  set_input_embeddingsmap_input_embeddingsr   r   )
r   r   r   r  r0   r  r  r  original_input_embeddingsr-  s
             r   rc   z$AssistantToTargetTranslator.__init__  ss    ;K=P!APA\'='=bg$&7335 	P+T-O /3.J.J.L @D'>'^?Z^C^$t''(1,++38<<;;BBDE**7740
 $3#H#H#J !78H$JjJj!k$55nE,;,P,P,R)'9:SUYUuUu'v$-445IJ,@))<243O3O3QSWSoSopq*&' -r   c                 @    | j                   rd| j                  _        yy)as  
        Disables the mapping of input ids despite the assistant pruning for the language model head being enabled.

        This method is required for the first forward pass of `_MapInputEmbedding` where input ids are already in the assistant vocabulary space. By disabling the mapping, it ensures that the input ids are processed correctly without remapping.

        FN)r  r-  r  r   s    r   unmap_input_idsz+AssistantToTargetTranslator.unmap_input_ids  s      '',1D%%) (r   c           	      V   | j                   j                         }| j                  j                         }d}| j                  |d      d   }t        |      dkD  r| j                   j	                  |      d   d   }| j                  |d      d   }t        |      dkD  rm| j                  j	                  |      d   d   }||k7  rG|j                         D 	ci c],  \  }}	|j                  |      r|j                  ||d      n||	. }}}	t        |j                               }
t        j                  |
dz   f| j                  t              }i }|j                         D ]#  \  }}|j                  |      }||||<   |||<   % |j                  | j                         |fS c c}	}w )N F)r   r   r   r   r<   )r   	get_vocabr!  rt   convert_ids_to_tokensrB   
startswithreplaceru   r   r(   fullr  r,   rP   r>   r"  )r   target_vocabassistant_vocab	space_strtarget_space_idstarget_space_signassistant_space_idsassistant_space_signtokr   max_assistant_indexassistant_to_target_input_idsr%  assistant_id	target_ids                  r   r#  z>AssistantToTargetTranslator._get_assistant_to_target_input_ids  s   --77933==?	11)PU1VWbc 1$ $ 6 6 L LM] ^_` abc d"&";";IZ_";"`al"m&'!+'+'@'@'V'VWj'klm'nop'q$$(<< )8(=(=(?' %C  #~~.BC  KK(<>OQRS!$'O ' "/"8"8":;(-

4G!4K3MtOeOemp(q%8:%!0!6!6!8 	HC$((-I$>G-l;;G-i8		H
 -//0L0LMOlll#'s   1F%r   c                 `    t        j                  | j                  | j                  k(        d   S )z`
        Get the input ids that are in the assistant vocab but not in the target vocab.
        r   )r(   wherer$  r  r0  s    r   r&  z3AssistantToTargetTranslator._get_suppress_input_ids  s*     {{4>>$BXBXXYZ[\\r   assistant_candidate_idsc                    t        |d         |j                  d   z
  }|dk(  r|S |d| df   }| j                  r| j                  |   }| j                  |   }t        j                  ||j                  d      fd      S )aS  
        Return the target candidate ids that correspond to the assistant candidate ids.
        Note that we have already the target ids for the prompt and we only need to find the target ids for the new tokens.
        Moreover, assistant ids of the original prompt does not necessarily appear in _assistant_to_target_input_ids.
        r   r   Nr   )rt   rR   r  r  r$  r(   r   r  )r   r   target_input_idsrH  num_new_tokenslast_candidate_idstransformed_slices          r   get_target_idsz*AssistantToTargetTranslator.get_target_ids  s     4Q78;N;T;TUV;WWQ## "9^O<L9L!M++%)%E%EFX%Y" $ C CDV W99.0A0K0KA0NOUVWWr   assistant_logitsc                 l   g |j                   dd | j                  }t        j                  || j                  | j
                        }| j                  | j                  k7  }| j                  |   }| j                  r	||d|f<   |S |dd| j                  j                   d   f   }|d|f   |d|f<   |S )zS
        Return the target logits that correspond to the assistant logits.
        Nr   r;   .r   )	rR   r  r(   r9  r  r"  r$  r  r  )r   rO  target_shapetarget_logitsassistant_indices_masktarget_logits_supported_indicesvalid_assistant_logitss          r   get_target_logitsz-AssistantToTargetTranslator.get_target_logits(  s    
 )_*:*@*@"*E(^tG]G](^+0::$++D4P4P,
 "&!D!DH^H^!^*.*M*MNd*e'''BRM#>>?  &6c;iT=`=`=f=fgh=i;i6i%j"BXY\^tYtBuM#>>?r   NF)r$   r%   r&   r'   floatr  __annotations__r  r,   r   r   rc   r1  r#  r   r&  r(   r)   rN  r+   rW  r-   r   r   r  r    s    ( !<-L%'s 8<(-'3' 7' 	'
 ""34' "&'R2!mF]c ]XNSN^N^X			X*%2C2C HYHY r   r  c                   x    e Zd ZdZ ej
                         Ze	 	 ddddddede	d   d	e
d
efd       Zed        Zy)AssistantVocabTranslatorCachez
    Cache for `AssistantToTargetTranslator` instances. The instances are computed at
    pre-processing time, and this cache allows us to avoid recomputing them.
    Nr   r   r   r  r0   r   r  r   c                     | j                   j                  |      }|#t        j                         }|| j                   |<   |j                  |      }|t	        |||||      }|||<   |S r   )_cacherP   weakrefWeakKeyDictionaryr  )clsr   r   r  r0   r  assistant_dictmappings           r   get_translatorz,AssistantVocabTranslatorCache.get_translatorF  s{     (89!$668N+9CJJ'( $$%89?1 #!'G 3:N./r   c                     | j                   D cg c]  }||	 }}|D ]  }| j                   |=  | j                   j                         D ]  }|D cg c]  }||	 }}|D ]  }||=    yc c}w c c}w )z
        Clean up dead references in the cache.
        This removes entries where either the target_tokenizer or assistant_tokenizer
        has been garbage collected.
        N)r^  r   )ra  r_   	dead_keysrb  s       r   cleanupz%AssistantVocabTranslatorCache.cleanupa  s     %(JJ>S#+S>	> 	 C

3	  "jj//1 	(N(6F#+FIF  ("3'(	( ? Gs   A-A-A2A2rX  )r$   r%   r&   r'   r_  r`  r^  classmethodr,   r   r   r  rd  rg  r-   r   r   r\  r\  >  s    
 'W&&(F 8<(-3 7 	
 ""34 "& 
% 4 ( (r   r\  c                   >    e Zd ZdZ	 	 ddej
                  dddddddd	d
ededeej                     ddf fdZ
dej
                  deej
                  eej                     f   fdZddej
                  dedef fdZdej
                  dej
                  fdZ xZS )%UniversalSpeculativeDecodingGeneratorz
    `CandidateGenerator` class to be used for Universal Speculative Decoding (USD): speculative decoding with different tokenizers
    for the assistant and main models. This class generates candidates through the use of a smaller model.
    r   r0   r   r   r   r   r1   r   r2   atm_translatorr3   r4   r   c
           
      \    || _         t        
| 	  ||||||||	       d| _        d | _        y )Nr   )_atm_translatorr   rc   _target_seq_len_with_candidates_prev_assistant_ids)r   r   r0   r   r   r1   r2   rk  r3   r4   r   s             r   rc   z.UniversalSpeculativeDecodingGenerator.__init__z  sD      .		
 56,?C r   r   c                 f   |j                  | j                  j                        }| j                  |      \  }}| j	                  |      \  }}|dk(  r|dfS | j                  ||       | j                  |||      }d|d   _        d|d   _        | j                  j                  | j                  j                  |d<   | j                  |      \  | _        }| j                  j                  ||| j                        }	|	j                  d   | _        | j                  j!                  |      }
|	|
fS )zk
        Simplified version of get_candidates that uses the translator cache for token conversion.
        r   Nr   Tr1   r4   r   )r>   r0   r;   r   rf   rg   rh   rU   rT   rm  r(  ri   ro  rN  rR   rn  rW  )r   r   rJ  r   r   rY   rj   rk   assistant_candidate_logitstarget_candidate_idstarget_candidate_logitss              r   r   z4UniversalSpeculativeDecodingGenerator.get_candidates  sO    %<<(<(<(C(CD040Q0QRb0c--)-)C)CDT)U&Qd?"##$7JZ#[778K^]kl >B+,:GK+,D 11=262F2F2X2XO./?C?X?XYh?i< "<  $33BB!143K3K 
 0D/I/I"/M,"&"6"6"H"HIc"d#%<<<r   r   r   c                     | j                   Gt        | j                  |j                  d   | j                  j
                  j                        | _        t        | !  ||      S )Nr   rq  )	ro  r   rN   rR   r0   rI   rJ   r   rg   )r   r   r   r   s      r   rg   z<UniversalSpeculativeDecodingGenerator._update_past_and_masks  sb    ##+ %<%%':'@'@'DdFZFZFaFaFtFt%D! w-.ATd-eer   rJ  c                 Z   |j                   d   }| j                  dk(  r|}nd}|dd| df   }d}| j                  dkD  r6| j                  j                  j	                  |d   j                               }|Y| j                  j                  |dd      }| j                  |dd	      d
   j                  | j                  j                        }n-t        j                  |gg| j                  j                        }| j                  |}nU| j                  dz   |z
  }|dkD  r| j                  ddd| f   | _        t        j                  | j                  |gd      }|j                  t        j                         }| j                  j#                          |t%        |d         fS )zM
        Simplified token conversion that only processes new tokens.
        r   r   r   NTr   Fr   r   r   rQ  r   r4  )rR   rn  rm  r%  rP   r   r   r   r   r>   r0   r;   r(   r   ro  r   rS   r1  rt   )	r   rJ  target_seq_lennew_token_counttarget_new_idsassistant_new_idstarget_new_textr   tokens_to_removes	            r   r   zBUniversalSpeculativeDecodingGenerator._prepare_assistant_input_ids  s   
 *//3//14,OO)!o-=->*>? !//!3 $ 4 4 R R V VWefgWhWmWmWo p$"33@@DW[ A O !% 8 8E$ !9 !!2d2299:  !&/@.A-B4K_K_KfKf g ##+"3#CCaG.X!#+/+C+CAGYIYHYGYDY+Z("'))T-E-EGX,Y_a"b1445::4F,,."C(9!(<$===r   r   )r   )r$   r%   r&   r'   r(   r)   r   r  r   rD   rc   r*   r+   r   r,   r   rg   r   r   r  s   @r   rj  rj  t  s    1526D##D +D 4	D
 7D .D D 4D  -D 0D8=(8(8 =U5CSCSU]^c^o^oUpCp=q =@f%:J:J f^a fjn f&>U=M=M &>RWRbRb &>r   rj  c            	           e Zd ZdZ	 	 	 	 ddeej                     dedee   defdZdej                  d	e
ej                  eej                     f   fd
Zdej                  dej                  defdZy)PromptLookupCandidateGeneratora
  
    `CandidateGenerator` class to be used for prompt lookup generation. This class generates candidates by looking up
    likely continuations in the provided prompt (input_ids) itself.
    Read the following blog post for more information: https://github.com/apoorvumang/prompt-lookup-decoding

    Args:
        max_matching_ngram_size (`int`):
            The maximum ngram size to be considered for matching in the prompt
        num_output_tokens (`int`):
            The number of tokens to be output as candidate tokens.
        max_length (`int`):
            The number of total maximum tokens that can be generated. For decoder-only models that includes the prompt length.
            Defaults to 20, which is the max length used as default in generation config.
    NrA   num_output_tokensmax_matching_ngram_sizer   c                     || _         |r|nd| _        || _        || _        | j                  dk  s| j                   dk  rt	        d      y )Nr   r   z4Invalid max_matching_ngram_size or num_output_tokens)r  r  r   rA   rZ   )r   rA   r  r  r   s        r   rc   z'PromptLookupCandidateGenerator.__init__  sV     "3BY'>_`$$(''1,0F0F!0KSTT 1Lr   r   r   c                    |j                  d      }| j                  |dz   k(  r|dfS d}d}t        t        | j                  |dz
        dd      D ]  }|j                  d|d      }|d| df   }||k(  j                  d      }|j                  d	
      d   }	|	D ]  }
|
|z   }|| j                  z   }t        ||| j                        }||k  s4|d||f   }d	}t        || j                        }t        j                  |      }|j                         dkD  r|d   j                         }|d| } n |s n |t        |      dk(  r|dfS |j                  d      }t        j                   ||fd      }|dfS )a  
        Fetches the candidates to be tried for the current input.

        Args:
            input_ids (`torch.LongTensor` of shape `(batch_size, sequence_length)`):
                Indices of input sequence tokens in the vocabulary. [What are input IDs?](../glossary#input-ids)

        Return:
            `torch.LongTensor` of shape `(num_candidates, candidate_length)`: The candidate sequences to be tried.
        r   NFr   r   )	dimensionsizestepr   r   T)as_tuple)r  r   r   r   r  unfoldallr   r  r
   rA   r(   numelr   rt   r  r   )r   r   input_length
chosen_idsmatch_found
ngram_sizewindowsngram_tensorr^   match_indicesr   r   end_idxmaskmatch_indices_eosfirst_eos_indexcandidate_input_idss                    r   r   z-PromptLookupCandidateGenerator.get_candidates  s    !~~a( ??lQ..d?"
D$@$@,QRBR SUVXZ[ !	J&&!&LG %Q_5L ,.333:G $OOTO:1=M % *,	#d&<&<<g|T__Ew&!*1i.?+?!@J"&K
 -Z9J9JKD(-d(;%(..014*;A*>*C*C*E%/0@%A
#$ C!	F ZA!5d?"  ))!,
#iiJ(?QG"D((r   r   r   c                      y)r!   Nr-   r"   s       r   r#   z8PromptLookupCandidateGenerator.update_candidate_strategyG  s     	r   )N
   N   )r$   r%   r&   r'   r   r(   rD   r,   rc   r)   r*   r+   r   r#   r-   r   r   r~  r~    s    " 04!#15Uu||,U U "*#	U
 U>)(8(8 >)U5CSCSU]^c^o^oUpCp=q >)@53C3C UM^M^ mp r   r~  c                        e Zd ZdZ	 	 ddej
                  dddddedeej                     d	d
f fdZ	dej
                  de
ej
                  eej                     f   f fdZ xZS )EarlyExitCandidateGeneratora  
    `CandidateGenerator` class to be used for assisted generation and speculative decoding. This class generates
    candidates through the use of **the model itself**, exiting early. Can only be used with models that support early
    exit, e.g., `facebook/layerskip-llama3.2-1B`.

    Args:
        input_ids (`torch.LongTensor` of shape `(batch_size, sequence_length)`):
            Indices of input sequence tokens in the vocabulary. [What are input IDs?](../glossary#input-ids)
        assistant_model (`PreTrainedModel`):
            The original model. This model must support early exit (i.e. is trained to compute logits in earlier
            layers).
        generation_config (`~generation.GenerationConfig`, *optional*):
            The generation configuration to be used as base parametrization for the generation call.
        logits_processor (`LogitsProcessorList`):
            An instance of [`LogitsProcessorList`]. List of instances of class derived from [`LogitsProcessor`]
            used to modify the prediction scores of the language modeling head applied at each generation step.
        model_kwargs (`Dict`):
            The keyword arguments that will be passed to the main model, and are used as base inputs for the assistant
            model as well.
        inputs_tensor (`torch.Tensor`, *optional*):
            The model input tensor. In encoder-decoder models, this is the encoder input.
    r   r0   r   r1   r   r2   r3   r4   r   c                     t         |   ||||||       | j                  j                  | _        d | j                  _        y )N)r   r0   r1   r2   r3   r4   )r   rc   r1   assistant_early_exit)r   r   r0   r1   r2   r3   r4   r   s          r   rc   z$EarlyExitCandidateGenerator.__init__p  sM     	+/%'- 	 	
 %)$:$:$O$O!6:3r   r   c                    t        | j                  | j                  j                        }|j                  j                  }| j
                  |j                  _        t        |   |      \  }}||j                  _        ||fS r   )getattrr0   base_model_prefixrI   num_hidden_layersr  r   r   )r   r   
base_modeloriginal_num_hidden_layersrl   rm   r   s         r   r   z*EarlyExitCandidateGenerator.get_candidates  sv    T1143G3G3Y3YZ
%/%6%6%H%H".2.G.G
+*/'*@*K''.H
+...r   r   )r$   r%   r&   r'   r(   r)   r   r   rD   rc   r*   r+   r   r   r  s   @r   r  r  X  s    : 1526;##; +; .	;
 ;  -; 0;,/(8(8 /U5CSCSU]^c^o^oUpCp=q / /r   r  r2   
new_lengthrJ   r   c                 v   |rdnd}|| vr| S | |   }||j                   d   z
  }|dk  r|ddd|f   | |<   n?|dkD  r:t        j                  ||j                  |j                   d   |f      gd      | |<   d| v r`| d   }|dk  r|ddd|f   | d<   | S |dkD  r?|ddddddddf   j	                  d|dd      }t        j                  ||gd      | d<   | S d	| v rZ| d	   }|dk  r|ddd|f   | d	<   | S |dkD  r;|ddddddf   j	                  d|d      }t        j                  ||gd      | d	<   | S )
zNExpands or crops the model's mask for decoding purposes, to the defined lengthr:   r=   r   r   Nr   r   cross_attention_maskimage_attention_mask)rR   r(   r   new_onesrepeat)r2   r  rJ   mask_keyr  mask_length_diff
cross_masknew_masks           r   r   r     s    ,>'CSH|#!D!DJJqM1!!%a):*:):&:!;X	A	!&D$--AP`@a2b+cik!lX -!"89
a3=aARBRAR>R3SL/0  !!!RS!Q,/66q:JAqQH3899j(=SYZ3[L/0  
 <	/!"89
a3=aARBRAR>R3SL/0
 	 !!!RS!),33A7GKH3899j(=SYZ3[L/0r   c                 
   d| vs| d   | S | d   }|dddf   j                  d      }||j                  d   z
  }|dk  r|ddd|f   }| S |dkD  r1|j                  d|      }t        j                  | d   |gd      | d<   | S )zXExpands or crops the model's token_type_ids for decoding purposes, to the defined lengthtoken_type_idsNr   r   r   r   )r  rR   r  r(   r   )r2   r  r  final_token_typetype_length_difftoken_type_copiess         r   r   r     s    |+|<L/M/U!"23N%ae,66r:!N$8$8$;;!'+<,<+<(<=  
A	,33A7GH).LAQ4RTe3fln)o%&r   )-rF   r_  typingr   r   r   numpyry   r(   torch.nnr  pytorch_utilsr   utilsr   sklearn.metricsr	   r
   logits_processr   r   r   modeling_utilsr   tokenization_utils_baser   configuration_utilsr   r   r/   r   Moduler  r  r  r\  rj  r~  r  r   strr,   r   r   r   r-   r   r   <module>r     s;      / /    . ( ) - h h 0A5$
 $
Ny/!3 y/xW4N WtRYY .5 5D\ \~3( 3(lq>,Y q>hm%7 m`5/"< 5/p!$sCx. !c !_c !hlmprumuhv !H$sCx. c dSVX[S[n r   