jason1966 commited on
Commit
2ace793
·
verified ·
1 Parent(s): 7ba78b7

Upload modeling_internvl_chat.py with huggingface_hub

Browse files
Files changed (1) hide show
  1. modeling_internvl_chat.py +390 -0
modeling_internvl_chat.py ADDED
@@ -0,0 +1,390 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # --------------------------------------------------------
2
+ # InternVL
3
+ # Copyright (c) 2024 OpenGVLab
4
+ # Licensed under The MIT License [see NOTICE for details]
5
+ # --------------------------------------------------------
6
+
7
+ import warnings
8
+ from typing import List, Optional, Tuple, Union
9
+
10
+ import torch.utils.checkpoint
11
+ import transformers
12
+ from torch import nn
13
+ from torch.nn import CrossEntropyLoss
14
+ from transformers import GenerationConfig
15
+ from transformers.modeling_outputs import CausalLMOutputWithPast
16
+ from transformers.modeling_utils import PreTrainedModel
17
+ from transformers.utils import logging
18
+ from transformers import LlamaForCausalLM, Qwen2ForCausalLM, Qwen3ForCausalLM, Qwen3MoeForCausalLM
19
+
20
+ from .configuration_internvl_chat import InternVLChatConfig
21
+ from .conversation import get_conv_template
22
+ from .modeling_intern_vit import InternVisionModel, has_flash_attn
23
+
24
+ logger = logging.get_logger(__name__)
25
+
26
+
27
+ def version_cmp(v1, v2, op='eq'):
28
+ import operator
29
+
30
+ from packaging import version
31
+ op_func = getattr(operator, op)
32
+ return op_func(version.parse(v1), version.parse(v2))
33
+
34
+
35
+ class InternVLChatModel(PreTrainedModel):
36
+ config_class = InternVLChatConfig
37
+ main_input_name = 'pixel_values'
38
+ base_model_prefix = 'language_model'
39
+ _supports_flash_attn_2 = True
40
+ supports_gradient_checkpointing = True
41
+ _no_split_modules = [
42
+ "InternVisionModel",
43
+ "Qwen3DecoderLayer",
44
+ ]
45
+
46
+ # support transformers 4.51.+
47
+ _tp_plan = ''
48
+
49
+ @property
50
+ def all_tied_weights_keys(self):
51
+ if hasattr(self, 'language_model'):
52
+ return getattr(
53
+ self.language_model,
54
+ 'all_tied_weights_keys',
55
+ getattr(self.language_model, '_tied_weights_keys', []),
56
+ )
57
+ return getattr(self, '_tied_weights_keys', [])
58
+
59
+ def __init__(self, config: InternVLChatConfig, vision_model=None, language_model=None, use_flash_attn=True):
60
+ super().__init__(config)
61
+
62
+ assert version_cmp(transformers.__version__, '4.37.0', 'ge')
63
+ image_size = config.force_image_size or config.vision_config.image_size
64
+ patch_size = config.vision_config.patch_size
65
+ self.patch_size = patch_size
66
+ self.select_layer = config.select_layer
67
+ self.template = config.template
68
+ self.num_image_token = int((image_size // patch_size) ** 2 * (config.downsample_ratio ** 2))
69
+ self.downsample_ratio = config.downsample_ratio
70
+ self.ps_version = config.ps_version
71
+ use_flash_attn = use_flash_attn if has_flash_attn else False
72
+ config.vision_config.use_flash_attn = True if use_flash_attn else False
73
+ config.llm_config._attn_implementation = 'flash_attention_2' if use_flash_attn else 'eager'
74
+
75
+ logger.info(f'num_image_token: {self.num_image_token}')
76
+ logger.info(f'ps_version: {self.ps_version}')
77
+ if vision_model is not None:
78
+ self.vision_model = vision_model
79
+ else:
80
+ self.vision_model = InternVisionModel(config.vision_config)
81
+ if language_model is not None:
82
+ self.language_model = language_model
83
+ else:
84
+ architecture: str = config.llm_config.architectures[0]
85
+ if architecture == 'LlamaForCausalLM':
86
+ self.language_model = LlamaForCausalLM(config.llm_config)
87
+ elif architecture == 'Qwen2ForCausalLM':
88
+ self.language_model = Qwen2ForCausalLM(config.llm_config)
89
+ elif architecture == 'Qwen3MoeForCausalLM':
90
+ self.language_model = Qwen3MoeForCausalLM(config.llm_config)
91
+ elif architecture == 'Qwen3ForCausalLM':
92
+ self.language_model = Qwen3ForCausalLM(config.llm_config)
93
+ else:
94
+ raise NotImplementedError(f'{architecture} is not implemented.')
95
+
96
+ vit_hidden_size = config.vision_config.hidden_size
97
+ llm_hidden_size = config.llm_config.hidden_size
98
+
99
+ self.mlp1 = nn.Sequential(
100
+ nn.LayerNorm(vit_hidden_size * int(1 / self.downsample_ratio) ** 2),
101
+ nn.Linear(vit_hidden_size * int(1 / self.downsample_ratio) ** 2, llm_hidden_size),
102
+ nn.GELU(),
103
+ nn.Linear(llm_hidden_size, llm_hidden_size)
104
+ )
105
+
106
+ self.img_context_token_id = None
107
+ self.conv_template = get_conv_template(self.template)
108
+ self.system_message = self.conv_template.system_message
109
+
110
+ def forward(
111
+ self,
112
+ pixel_values: torch.FloatTensor,
113
+ input_ids: torch.LongTensor = None,
114
+ attention_mask: Optional[torch.Tensor] = None,
115
+ position_ids: Optional[torch.LongTensor] = None,
116
+ image_flags: Optional[torch.LongTensor] = None,
117
+ past_key_values: Optional[List[torch.FloatTensor]] = None,
118
+ labels: Optional[torch.LongTensor] = None,
119
+ use_cache: Optional[bool] = None,
120
+ output_attentions: Optional[bool] = None,
121
+ output_hidden_states: Optional[bool] = None,
122
+ return_dict: Optional[bool] = None,
123
+ ) -> Union[Tuple, CausalLMOutputWithPast]:
124
+ return_dict = return_dict if return_dict is not None else self.config.use_return_dict
125
+
126
+ image_flags = image_flags.squeeze(-1)
127
+ input_embeds = self.language_model.get_input_embeddings()(input_ids).clone()
128
+
129
+ vit_embeds = self.extract_feature(pixel_values)
130
+ vit_embeds = vit_embeds[image_flags == 1]
131
+ vit_batch_size = pixel_values.shape[0]
132
+
133
+ B, N, C = input_embeds.shape
134
+ input_embeds = input_embeds.reshape(B * N, C)
135
+
136
+ # if torch.distributed.is_initialized() and torch.distributed.get_rank() == 0:
137
+ # print(f'dynamic ViT batch size: {vit_batch_size}, images per sample: {vit_batch_size / B}, dynamic token length: {N}')
138
+
139
+ input_ids = input_ids.reshape(B * N)
140
+ selected = (input_ids == self.img_context_token_id)
141
+ try:
142
+ input_embeds[selected] = input_embeds[selected] * 0.0 + vit_embeds.reshape(-1, C)
143
+ except Exception as e:
144
+ vit_embeds = vit_embeds.reshape(-1, C)
145
+ print(f'warning: {e}, input_embeds[selected].shape={input_embeds[selected].shape}, '
146
+ f'vit_embeds.shape={vit_embeds.shape}')
147
+ n_token = min(selected.sum(), vit_embeds.size(0))
148
+ input_embeds[selected][:n_token] = input_embeds[selected][:n_token] * 0.0 + vit_embeds[:n_token]
149
+
150
+ input_embeds = input_embeds.reshape(B, N, C)
151
+
152
+ outputs = self.language_model(
153
+ inputs_embeds=input_embeds,
154
+ attention_mask=attention_mask,
155
+ position_ids=position_ids,
156
+ past_key_values=past_key_values,
157
+ use_cache=use_cache,
158
+ output_attentions=output_attentions,
159
+ output_hidden_states=output_hidden_states,
160
+ return_dict=return_dict,
161
+ )
162
+ logits = outputs.logits
163
+
164
+ loss = None
165
+ if labels is not None:
166
+ # Shift so that tokens < n predict n
167
+ shift_logits = logits[..., :-1, :].contiguous()
168
+ shift_labels = labels[..., 1:].contiguous()
169
+ # Flatten the tokens
170
+ loss_fct = CrossEntropyLoss()
171
+ shift_logits = shift_logits.view(-1, self.language_model.config.vocab_size)
172
+ shift_labels = shift_labels.view(-1)
173
+ # Enable model parallelism
174
+ shift_labels = shift_labels.to(shift_logits.device)
175
+ loss = loss_fct(shift_logits, shift_labels)
176
+
177
+ if not return_dict:
178
+ output = (logits,) + outputs[1:]
179
+ return (loss,) + output if loss is not None else output
180
+
181
+ return CausalLMOutputWithPast(
182
+ loss=loss,
183
+ logits=logits,
184
+ past_key_values=outputs.past_key_values,
185
+ hidden_states=outputs.hidden_states,
186
+ attentions=outputs.attentions,
187
+ )
188
+
189
+ def pixel_shuffle(self, x, scale_factor=0.5):
190
+ n, w, h, c = x.size()
191
+ # N, W, H, C --> N, W, H * scale, C // scale
192
+ x = x.view(n, w, int(h * scale_factor), int(c / scale_factor))
193
+ # N, W, H * scale, C // scale --> N, H * scale, W, C // scale
194
+ x = x.permute(0, 2, 1, 3).contiguous()
195
+ # N, H * scale, W, C // scale --> N, H * scale, W * scale, C // (scale ** 2)
196
+ x = x.view(n, int(h * scale_factor), int(w * scale_factor),
197
+ int(c / (scale_factor * scale_factor)))
198
+ if self.ps_version == 'v1':
199
+ warnings.warn("In ps_version 'v1', the height and width have not been swapped back, "
200
+ 'which results in a transposed image.')
201
+ else:
202
+ x = x.permute(0, 2, 1, 3).contiguous()
203
+ return x
204
+
205
+ def extract_feature(self, pixel_values):
206
+ if self.select_layer == -1:
207
+ vit_embeds = self.vision_model(
208
+ pixel_values=pixel_values,
209
+ output_hidden_states=False,
210
+ return_dict=True).last_hidden_state
211
+ else:
212
+ vit_embeds = self.vision_model(
213
+ pixel_values=pixel_values,
214
+ output_hidden_states=True,
215
+ return_dict=True).hidden_states[self.select_layer]
216
+ vit_embeds = vit_embeds[:, 1:, :]
217
+
218
+ h = w = int(vit_embeds.shape[1] ** 0.5)
219
+ vit_embeds = vit_embeds.reshape(vit_embeds.shape[0], h, w, -1)
220
+ vit_embeds = self.pixel_shuffle(vit_embeds, scale_factor=self.downsample_ratio)
221
+ vit_embeds = vit_embeds.reshape(vit_embeds.shape[0], -1, vit_embeds.shape[-1])
222
+ vit_embeds = self.mlp1(vit_embeds)
223
+ return vit_embeds
224
+
225
+ def batch_chat(self, tokenizer, pixel_values, questions, generation_config=None, num_patches_list=None,
226
+ history=None, return_history=False, IMG_START_TOKEN='<img>', IMG_END_TOKEN='</img>',
227
+ IMG_CONTEXT_TOKEN='<IMG_CONTEXT>', verbose=False, image_counts=None):
228
+ if history is not None or return_history:
229
+ print('Now multi-turn chat is not supported in batch_chat.')
230
+ raise NotImplementedError
231
+
232
+ if image_counts is not None:
233
+ num_patches_list = image_counts
234
+ print('Warning: `image_counts` is deprecated. Please use `num_patches_list` instead.')
235
+
236
+ img_context_token_id = tokenizer.convert_tokens_to_ids(IMG_CONTEXT_TOKEN)
237
+ self.img_context_token_id = img_context_token_id
238
+
239
+ if verbose and pixel_values is not None:
240
+ image_bs = pixel_values.shape[0]
241
+ print(f'dynamic ViT batch size: {image_bs}')
242
+
243
+ queries = []
244
+ for idx, num_patches in enumerate(num_patches_list):
245
+ question = questions[idx]
246
+ if pixel_values is not None and '<image>' not in question:
247
+ question = '<image>\n' + question
248
+ template = get_conv_template(self.template)
249
+ template.system_message = self.system_message
250
+ template.append_message(template.roles[0], question)
251
+ template.append_message(template.roles[1], None)
252
+ query = template.get_prompt()
253
+
254
+ image_tokens = IMG_START_TOKEN + IMG_CONTEXT_TOKEN * self.num_image_token * num_patches + IMG_END_TOKEN
255
+ query = query.replace('<image>', image_tokens, 1)
256
+ queries.append(query)
257
+
258
+ tokenizer.padding_side = 'left'
259
+ model_inputs = tokenizer(queries, return_tensors='pt', padding=True)
260
+ input_ids = model_inputs['input_ids'].to(self.device)
261
+ attention_mask = model_inputs['attention_mask'].to(self.device)
262
+ eos_token_id = tokenizer.convert_tokens_to_ids(template.sep.strip())
263
+ if generation_config is None:
264
+ generation_config = {}
265
+ generation_config['eos_token_id'] = eos_token_id
266
+ generation_output = self.generate(
267
+ pixel_values=pixel_values,
268
+ input_ids=input_ids,
269
+ attention_mask=attention_mask,
270
+ **generation_config
271
+ )
272
+ responses = tokenizer.batch_decode(generation_output, skip_special_tokens=False)
273
+ responses = [response.split(template.sep.strip())[0].strip() for response in responses]
274
+ return responses
275
+
276
+ def chat(self, tokenizer, pixel_values, question, generation_config=None, history=None, return_history=False,
277
+ num_patches_list=None, IMG_START_TOKEN='<img>', IMG_END_TOKEN='</img>', IMG_CONTEXT_TOKEN='<IMG_CONTEXT>',
278
+ verbose=False):
279
+
280
+ if history is None and pixel_values is not None and '<image>' not in question:
281
+ question = '<image>\n' + question
282
+
283
+ if num_patches_list is None:
284
+ num_patches_list = [pixel_values.shape[0]] if pixel_values is not None else []
285
+ assert pixel_values is None or len(pixel_values) == sum(num_patches_list)
286
+
287
+ img_context_token_id = tokenizer.convert_tokens_to_ids(IMG_CONTEXT_TOKEN)
288
+ self.img_context_token_id = img_context_token_id
289
+
290
+ template = get_conv_template(self.template)
291
+ template.system_message = self.system_message
292
+ eos_token_id = tokenizer.convert_tokens_to_ids(template.sep.strip())
293
+
294
+ history = [] if history is None else history
295
+ for (old_question, old_answer) in history:
296
+ template.append_message(template.roles[0], old_question)
297
+ template.append_message(template.roles[1], old_answer)
298
+ template.append_message(template.roles[0], question)
299
+ template.append_message(template.roles[1], None)
300
+ query = template.get_prompt()
301
+
302
+ if verbose and pixel_values is not None:
303
+ image_bs = pixel_values.shape[0]
304
+ print(f'dynamic ViT batch size: {image_bs}')
305
+
306
+ for num_patches in num_patches_list:
307
+ image_tokens = IMG_START_TOKEN + IMG_CONTEXT_TOKEN * self.num_image_token * num_patches + IMG_END_TOKEN
308
+ query = query.replace('<image>', image_tokens, 1)
309
+
310
+ model_inputs = tokenizer(query, return_tensors='pt')
311
+ input_ids = model_inputs['input_ids'].to(self.device)
312
+ attention_mask = model_inputs['attention_mask'].to(self.device)
313
+ if generation_config is None:
314
+ generation_config = {}
315
+ generation_config['eos_token_id'] = eos_token_id
316
+ generation_output = self.generate(
317
+ pixel_values=pixel_values,
318
+ input_ids=input_ids,
319
+ attention_mask=attention_mask,
320
+ **generation_config
321
+ )
322
+ response = tokenizer.batch_decode(generation_output, skip_special_tokens=False)[0]
323
+ response = response.split(template.sep.strip())[0].strip()
324
+ history.append((question, response))
325
+ if return_history:
326
+ return response, history
327
+ else:
328
+ query_to_print = query.replace(IMG_CONTEXT_TOKEN, '')
329
+ query_to_print = query_to_print.replace(f'{IMG_START_TOKEN}{IMG_END_TOKEN}', '<image>')
330
+ if verbose:
331
+ print(query_to_print, response)
332
+ return response
333
+
334
+ @torch.no_grad()
335
+ def generate(
336
+ self,
337
+ pixel_values: Optional[torch.FloatTensor] = None,
338
+ input_ids: Optional[torch.FloatTensor] = None,
339
+ attention_mask: Optional[torch.LongTensor] = None,
340
+ visual_features: Optional[torch.FloatTensor] = None,
341
+ generation_config: Optional[GenerationConfig] = None,
342
+ output_hidden_states: Optional[bool] = None,
343
+ **generate_kwargs,
344
+ ) -> torch.LongTensor:
345
+
346
+ assert self.img_context_token_id is not None
347
+ if pixel_values is not None:
348
+ if visual_features is not None:
349
+ vit_embeds = visual_features
350
+ else:
351
+ vit_embeds = self.extract_feature(pixel_values)
352
+ input_embeds = self.language_model.get_input_embeddings()(input_ids)
353
+ B, N, C = input_embeds.shape
354
+ input_embeds = input_embeds.reshape(B * N, C)
355
+
356
+ input_ids = input_ids.reshape(B * N)
357
+ selected = (input_ids == self.img_context_token_id)
358
+ assert selected.sum() != 0
359
+ input_embeds[selected] = vit_embeds.reshape(-1, C).to(input_embeds.device)
360
+
361
+ input_embeds = input_embeds.reshape(B, N, C)
362
+ else:
363
+ input_embeds = self.language_model.get_input_embeddings()(input_ids)
364
+
365
+ outputs = self.language_model.generate(
366
+ inputs_embeds=input_embeds,
367
+ attention_mask=attention_mask,
368
+ generation_config=generation_config,
369
+ output_hidden_states=output_hidden_states,
370
+ use_cache=True,
371
+ **generate_kwargs,
372
+ )
373
+
374
+ return outputs
375
+
376
+ @property
377
+ def lm_head(self):
378
+ return self.language_model.get_output_embeddings()
379
+
380
+ def get_output_embeddings(self):
381
+ return self.language_model.get_output_embeddings()
382
+
383
+ def get_input_embeddings(self):
384
+ return self.language_model.get_input_embeddings()
385
+
386
+ def set_input_embeddings(self, value):
387
+ return self.language_model.set_input_embeddings(value)
388
+
389
+ def set_output_embeddings(self, value):
390
+ return self.language_model.set_output_embeddings(value)