diff --git a/io_scene_fbx/__init__.py b/io_scene_fbx/__init__.py
index 19de480357ab8f27d62c71b24e85c0639782c6bb..ff6d0602a491504074e732dd6301809ca4a57add 100644
--- a/io_scene_fbx/__init__.py
+++ b/io_scene_fbx/__init__.py
@@ -35,6 +35,8 @@ bl_info = {
 
 if "bpy" in locals():
     import imp
+    if "import_fbx" in locals():
+        imp.reload(import_fbx)
     if "export_fbx" in locals():
         imp.reload(export_fbx)
 
@@ -46,11 +48,78 @@ from bpy.props import (StringProperty,
                        EnumProperty,
                        )
 
-from bpy_extras.io_utils import (ExportHelper,
+from bpy_extras.io_utils import (ImportHelper,
+                                 ExportHelper,
                                  path_reference_mode,
                                  axis_conversion,
                                  )
 
+class ImportFBX(bpy.types.Operator, ImportHelper):
+    """Load a FBX geometry file"""
+    bl_idname = "import_scene.fbx"
+    bl_label = "Import FBX"
+    bl_options = {'UNDO'}
+
+    directory = StringProperty()
+
+    filename_ext = ".fbx"
+    filter_glob = StringProperty(default="*.fbx", options={'HIDDEN'})
+
+    use_image_search = BoolProperty(
+            name="Image Search",
+            description="Search subdirs for any associated images "
+                        "(Warning, may be slow)",
+            default=True,
+            )
+
+    axis_forward = EnumProperty(
+            name="Forward",
+            items=(('X', "X Forward", ""),
+                   ('Y', "Y Forward", ""),
+                   ('Z', "Z Forward", ""),
+                   ('-X', "-X Forward", ""),
+                   ('-Y', "-Y Forward", ""),
+                   ('-Z', "-Z Forward", ""),
+                   ),
+            default='-Z',
+            )
+    axis_up = EnumProperty(
+            name="Up",
+            items=(('X', "X Up", ""),
+                   ('Y', "Y Up", ""),
+                   ('Z', "Z Up", ""),
+                   ('-X', "-X Up", ""),
+                   ('-Y', "-Y Up", ""),
+                   ('-Z', "-Z Up", ""),
+                   ),
+            default='Y',
+            )
+    global_scale = FloatProperty(
+            name="Scale",
+            min=0.001, max=1000.0,
+            default=1.0,
+            )
+
+    def execute(self, context):
+        from mathutils import Matrix
+
+        keywords = self.as_keywords(ignore=("axis_forward",
+                                            "axis_up",
+                                            "global_scale",
+                                            "filter_glob",
+                                            "directory",
+                                            ))
+
+        global_matrix = (Matrix.Scale(self.global_scale, 4) *
+                         axis_conversion(from_forward=self.axis_forward,
+                                         from_up=self.axis_up,
+                                         ).to_4x4())
+        keywords["global_matrix"] = global_matrix
+        keywords["use_cycles"] = (context.scene.render.engine == 'CYCLES')
+
+        from . import import_fbx
+        return import_fbx.load(self, context, **keywords)
+
 
 class ExportFBX(bpy.types.Operator, ExportHelper):
     """Selection to an ASCII Autodesk FBX"""
@@ -74,7 +143,7 @@ class ExportFBX(bpy.types.Operator, ExportHelper):
             description=("Scale all data "
                          "(Some importers do not support scaled armatures!)"),
             min=0.01, max=1000.0,
-            soft_min=0.01, soft_max=1000.0,
+            soft_min=0.001, soft_max=1000.0,
             default=1.0,
             )
     axis_forward = EnumProperty(
@@ -174,11 +243,6 @@ class ExportFBX(bpy.types.Operator, ExportHelper):
             description="Disable global rotation, for XNA compatibility",
             default=False,
             )
-    xna_validate = BoolProperty(
-            name="XNA Strict Options",
-            description="Make sure options are compatible with Microsoft XNA",
-            default=False,
-            )
     batch_mode = EnumProperty(
             name="Batch Mode",
             items=(('OFF', "Off", "Active scene to file"),
@@ -197,69 +261,26 @@ class ExportFBX(bpy.types.Operator, ExportHelper):
             options={'HIDDEN'},
             )
 
-    # Validate that the options are compatible with XNA (JCB)
-    def _validate_xna_options(self):
-        if not self.xna_validate:
-            return False
-        changed = False
-        if not self.use_rotate_workaround:
-            changed = True
-            self.use_rotate_workaround = True
-        if self.global_scale != 1.0:
-            changed = True
-            self.global_scale = 1.0
-        if self.mesh_smooth_type != 'OFF':
-            changed = True
-            self.mesh_smooth_type = 'OFF'
-        if self.use_anim_optimize:
-            changed = True
-            self.use_anim_optimize = False
-        if self.use_mesh_edges:
-            changed = True
-            self.use_mesh_edges = False
-        if self.use_default_take:
-            changed = True
-            self.use_default_take = False
-        if self.object_types & {'CAMERA', 'LAMP', 'EMPTY'}:
-            changed = True
-            self.object_types -= {'CAMERA', 'LAMP', 'EMPTY'}
-        if self.path_mode != 'STRIP':
-            changed = True
-            self.path_mode = 'STRIP'
-        return changed
-
     @property
     def check_extension(self):
         return self.batch_mode == 'OFF'
 
-    def check(self, context):
-        is_def_change = super().check(context)
-        is_xna_change = self._validate_xna_options()
-        return (is_xna_change or is_def_change)
-
     def execute(self, context):
         from mathutils import Matrix
         if not self.filepath:
             raise Exception("filepath not set")
 
-        global_matrix = Matrix()
 
-        global_matrix[0][0] = \
-        global_matrix[1][1] = \
-        global_matrix[2][2] = self.global_scale
-
-        if not self.use_rotate_workaround:
-            global_matrix = (global_matrix *
-                             axis_conversion(to_forward=self.axis_forward,
-                                             to_up=self.axis_up,
-                                             ).to_4x4())
+        global_matrix = (Matrix.Scale(self.global_scale, 4) *
+                         axis_conversion(to_forward=self.axis_forward,
+                                         to_up=self.axis_up,
+                                         ).to_4x4())
 
         keywords = self.as_keywords(ignore=("axis_forward",
                                             "axis_up",
                                             "global_scale",
                                             "check_existing",
                                             "filter_glob",
-                                            "xna_validate",
                                             ))
 
         keywords["global_matrix"] = global_matrix
@@ -268,20 +289,26 @@ class ExportFBX(bpy.types.Operator, ExportHelper):
         return export_fbx.save(self, context, **keywords)
 
 
-def menu_func(self, context):
+def menu_func_import(self, context):
+    self.layout.operator(ImportFBX.bl_idname, text="Autodesk FBX (.fbx)")
+
+
+def menu_func_export(self, context):
     self.layout.operator(ExportFBX.bl_idname, text="Autodesk FBX (.fbx)")
 
 
 def register():
     bpy.utils.register_module(__name__)
 
-    bpy.types.INFO_MT_file_export.append(menu_func)
+    bpy.types.INFO_MT_file_import.append(menu_func_import)
+    bpy.types.INFO_MT_file_export.append(menu_func_export)
 
 
 def unregister():
     bpy.utils.unregister_module(__name__)
 
-    bpy.types.INFO_MT_file_export.remove(menu_func)
+    bpy.types.INFO_MT_file_import.remove(menu_func_import)
+    bpy.types.INFO_MT_file_export.remove(menu_func_export)
 
 if __name__ == "__main__":
     register()
diff --git a/io_scene_fbx/export_fbx.py b/io_scene_fbx/export_fbx.py
index da833bad4908e6bb29fbae410e78e26b201269d3..8e1ad11e7e05930987ebcbe7a0e60f66e0501198 100644
--- a/io_scene_fbx/export_fbx.py
+++ b/io_scene_fbx/export_fbx.py
@@ -3050,19 +3050,6 @@ def save(operator, context,
 # Please update the lists for UDK, Unity, XNA etc. on the following web page:
 #   http://wiki.blender.org/index.php/Dev:2.5/Py/Scripts/Import-Export/UnifiedFBX
 
-# XNA FBX Requirements (JCB 29 July 2011)
-# - Armature must be parented to the scene
-# - Armature must be a 'Limb' never a 'null'.  This is in several places.
-# - First bone must be parented to the armature.
-# - Rotation must be completely disabled including
-#       always returning the original matrix in In object_tx().
-#       It is the animation that gets distorted during rotation!
-# - Lone edges cause intermittent errors in the XNA content pipeline!
-#       I have added a warning message and excluded them.
-# - Bind pose must be included with the 'MESH'
-# Typical settings for XNA export
-#   No Cameras, No Lamps, No Edges, No face smoothing, No Default_Take, Armature as bone, Disable rotation
-
 # NOTE TO Campbell -
 #   Can any or all of the following notes be removed because some have been here for a long time? (JCB 27 July 2011)
 # NOTES (all line numbers correspond to original export_fbx.py (under release/scripts)
diff --git a/io_scene_fbx/import_fbx.py b/io_scene_fbx/import_fbx.py
new file mode 100644
index 0000000000000000000000000000000000000000..0ab11f8ed047ccd4978d3daffc7a62c765fc7516
--- /dev/null
+++ b/io_scene_fbx/import_fbx.py
@@ -0,0 +1,566 @@
+# ##### BEGIN GPL LICENSE BLOCK #####
+#
+#  This program is free software; you can redistribute it and/or
+#  modify it under the terms of the GNU General Public License
+#  as published by the Free Software Foundation; either version 2
+#  of the License, or (at your option) any later version.
+#
+#  This program is distributed in the hope that it will be useful,
+#  but WITHOUT ANY WARRANTY; without even the implied warranty of
+#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+#  GNU General Public License for more details.
+#
+#  You should have received a copy of the GNU General Public License
+#  along with this program; if not, write to the Free Software Foundation,
+#  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+#
+# ##### END GPL LICENSE BLOCK #####
+
+# <pep8 compliant>
+
+# Script copyright (C) Blender Foundation
+
+# FBX 7.1.0 -> 7.3.0 loader for Blender
+
+import bpy
+
+# -----
+# Utils
+from .parse_fbx import data_types
+
+
+def tuple_deg_to_rad(eul):
+    return eul[0] / 57.295779513, eul[1] / 57.295779513, eul[2] / 57.295779513
+
+
+def elem_find_first(elem, id_search):
+    for fbx_item in elem.elems:
+        if fbx_item.id == id_search:
+            return fbx_item
+
+
+def elem_find_first_string(elem, id_search):
+    fbx_item = elem_find_first(elem, id_search)
+    if fbx_item is not None:
+        assert(len(fbx_item.props) == 1)
+        assert(fbx_item.props_type[0] == data_types.STRING)
+        return fbx_item.props[0].decode('utf-8')
+    return None
+
+
+def elem_find_first_bytes(elem, id_search, decode=True):
+    fbx_item = elem_find_first(elem, id_search)
+    if fbx_item is not None:
+        assert(len(fbx_item.props) == 1)
+        assert(fbx_item.props_type[0] == data_types.STRING)
+        return fbx_item.props[0]
+    return None
+
+
+def elem_repr(elem):
+    return "%s: props[%d=%r], elems=(%r)" % (
+        elem.id,
+        len(elem.props),
+        ", ".join([repr(p) for p in elem.props]),
+        # elem.props_type,
+        b", ".join([e.id for e in elem.elems]),
+        )
+
+
+def elem_split_name_class(elem):
+    """ Return
+    """
+    assert(elem.props_type[-2] == data_types.STRING)
+    elem_name, elem_class = elem.props[-2].split(b'\x00\x01')
+    return elem_name, elem_class
+
+
+def elem_uuid(elem):
+    assert(elem.props_type[0] == data_types.INT64)
+    return elem.props[0]
+
+
+def elem_prop_first(elem):
+    return elem.props[0] if (elem is not None) and elem.props else None
+
+
+# ----
+# Support for
+# Properties70: { ... P:
+def elem_props_find_first(elem, elem_prop_id):
+    for subelem in elem.elems:
+        assert(subelem.id == b'P')
+        if subelem.props[0] == elem_prop_id:
+            return subelem
+    return None
+
+
+def elem_props_get_color_rgb(elem, elem_prop_id, default=None):
+    elem_prop = elem_props_find_first(elem, elem_prop_id)
+    if elem_prop is not None:
+        assert(elem_prop.props[0] == elem_prop_id)
+        if elem_prop.props[1] == b'Color':
+            # FBX version 7300
+            assert(elem_prop.props[1] == b'Color')
+            assert(elem_prop.props[2] == b'')
+            assert(elem_prop.props[3] == b'A')
+        else:
+            assert(elem_prop.props[1] == b'ColorRGB')
+            assert(elem_prop.props[2] == b'Color')
+            #print(elem_prop.props_type[4:7])
+        assert(elem_prop.props_type[4:7] == bytes((data_types.FLOAT64,)) * 3)
+        return elem_prop.props[4:7]
+    return default
+
+
+def elem_props_get_number(elem, elem_prop_id, default=None):
+    elem_prop = elem_props_find_first(elem, elem_prop_id)
+    if elem_prop is not None:
+        assert(elem_prop.props[0] == elem_prop_id)
+        if elem_prop.props[1] == b'double':
+            assert(elem_prop.props[1] == b'double')
+            assert(elem_prop.props[2] == b'Number')
+        else:
+            assert(elem_prop.props[1] == b'Number')
+            assert(elem_prop.props[2] == b'')
+            assert(elem_prop.props[3] == b'A')
+
+        # we could allow other number types
+        assert(elem_prop.props_type[4] == data_types.FLOAT64)
+
+        return elem_prop.props[4]
+    return default
+
+
+# ----------------------------------------------------------------------------
+# Blender
+
+# ------
+# Object
+
+def blen_read_object(fbx_obj, object_data):
+    elem_name, elem_class = elem_split_name_class(fbx_obj)
+    elem_name_utf8 = elem_name.decode('utf-8')
+
+    const_vector_zero_3d = 0.0, 0.0, 0.0
+    const_vector_one_3d = 1.0, 1.0, 1.0
+
+    # Object data must be created already
+    obj = bpy.data.objects.new(name=elem_name_utf8, object_data=object_data)
+
+    fbx_props = elem_find_first(fbx_obj, b'Properties70')
+    assert(fbx_props is not None)
+
+    loc = elem_props_get_color_rgb(fbx_props, b'Lcl Translation', const_vector_zero_3d)
+    rot = elem_props_get_color_rgb(fbx_props, b'Lcl Rotation', const_vector_zero_3d)
+    sca = elem_props_get_color_rgb(fbx_props, b'Lcl Scaling', const_vector_one_3d)
+
+    obj.location = loc
+    obj.rotation_euler = tuple_deg_to_rad(rot)
+    obj.scale = sca
+
+    return obj
+
+
+# ----
+# Mesh
+
+def blen_read_geom_layerinfo(fbx_layer):
+    return (
+        elem_find_first_string(fbx_layer, b'Name'),
+        elem_find_first_bytes(fbx_layer, b'MappingInformationType'),
+        elem_find_first_bytes(fbx_layer, b'ReferenceInformationType'),
+        )
+
+
+def blen_read_geom_uv(fbx_obj, mesh):
+
+    for uvlayer_id in (b'LayerElementUV',):
+        fbx_uvlayer = elem_find_first(fbx_obj, uvlayer_id)
+
+        if fbx_uvlayer is None:
+            continue
+
+        # all should be valid
+        (fbx_uvlayer_name,
+         fbx_uvlayer_mapping,
+         fbx_uvlayer_ref,
+         ) = blen_read_geom_layerinfo(fbx_uvlayer)
+
+        # print(fbx_uvlayer_name, fbx_uvlayer_mapping, fbx_uvlayer_ref)
+
+        fbx_layer_data = elem_prop_first(elem_find_first(fbx_uvlayer, b'UV'))
+        fbx_layer_index = elem_prop_first(elem_find_first(fbx_uvlayer, b'UVIndex'))
+
+        # TODO, generic mappuing apply function
+        if fbx_uvlayer_mapping == b'ByPolygonVertex':
+            if fbx_uvlayer_ref == b'IndexToDirect':
+                # TODO, more generic support for mapping types
+                uv_tex = mesh.uv_textures.new(name=fbx_uvlayer_name)
+                uv_lay = mesh.uv_layers[fbx_uvlayer_name]
+                uv_data = [luv.uv for luv in uv_lay.data]
+
+                for i, j in enumerate(fbx_layer_index):
+                    uv_data[i][:] = fbx_layer_data[(j * 2): (j * 2) + 2]
+            else:
+                print("warning uv layer ref type unsupported:", fbx_uvlayer_ref)
+        else:
+            print("warning uv layer mapping type unsupported:", fbx_uvlayer_mapping)
+
+
+def blen_read_geom(fbx_obj):
+    elem_name, elem_class = elem_split_name_class(fbx_obj)
+    assert(elem_class == b'Geometry')
+    elem_name_utf8 = elem_name.decode('utf-8')
+
+    fbx_verts = elem_prop_first(elem_find_first(fbx_obj, b'Vertices'))
+    fbx_polys = elem_prop_first(elem_find_first(fbx_obj, b'PolygonVertexIndex'))
+    # TODO
+    # fbx_edges = elem_prop_first(elem_find_first(fbx_obj, b'Edges'))
+
+    mesh = bpy.data.meshes.new(name=elem_name_utf8)
+    mesh.vertices.add(len(fbx_verts) // 3)
+    mesh.vertices.foreach_set("co", fbx_verts)
+
+    mesh.loops.add(len(fbx_polys))
+
+    #poly_loops = []  # pairs (loop_start, loop_total)
+    poly_loop_starts = []
+    poly_loop_totals = []
+    poly_loop_prev = 0
+    for i, l in enumerate(mesh.loops):
+        index = fbx_polys[i]
+        if index < 0:
+            poly_loop_starts.append(poly_loop_prev)
+            poly_loop_totals.append((i - poly_loop_prev) + 1)
+            poly_loop_prev = i + 1
+            index = -(index + 1)
+        l.vertex_index = index
+    poly_loop_starts.append(poly_loop_prev)
+    poly_loop_totals.append((i - poly_loop_prev) + 1)
+
+    mesh.polygons.add(len(poly_loop_starts))
+    mesh.polygons.foreach_set("loop_start", poly_loop_starts)
+    mesh.polygons.foreach_set("loop_total", poly_loop_totals)
+
+    blen_read_geom_uv(fbx_obj, mesh)
+
+    mesh.validate()
+    mesh.calc_normals()
+
+    return mesh
+
+
+# --------
+# Material
+
+def blen_read_material(fbx_obj,
+                       cycles_material_wrap_map, use_cycles):
+    elem_name, elem_class = elem_split_name_class(fbx_obj)
+    assert(elem_class == b'Material')
+    elem_name_utf8 = elem_name.decode('utf-8')
+
+    ma = bpy.data.materials.new(name=elem_name_utf8)
+
+    const_color_white = 1.0, 1.0, 1.0
+
+    fbx_props = elem_find_first(fbx_obj, b'Properties70')
+    assert(fbx_props is not None)
+
+    ma_diff = elem_props_get_color_rgb(fbx_props, b'DiffuseColor', const_color_white)
+    ma_spec = elem_props_get_color_rgb(fbx_props, b'SpecularColor', const_color_white)
+    ma_alpha = elem_props_get_number(fbx_props, b'Opacity', 1.0)
+    ma_spec_intensity = ma.specular_intensity = elem_props_get_number(fbx_props, b'SpecularFactor', 0.25) * 2.0
+    ma_spec_hardness = elem_props_get_number(fbx_props, b'Shininess', 9.6)
+    ma_refl_factor = elem_props_get_number(fbx_props, b'ReflectionFactor', 0.0)
+    ma_refl_color = elem_props_get_color_rgb(fbx_props, b'ReflectionColor', const_color_white)
+
+    if use_cycles:
+        from . import cycles_shader_compat
+        # viewport color
+        ma.diffuse_color = ma_diff
+
+        ma_wrap = cycles_shader_compat.CyclesShaderWrapper(ma)
+        ma_wrap.diffuse_color_set(ma_diff)
+        ma_wrap.specular_color_set([c * ma_spec_intensity for c in ma_spec])
+        ma_wrap.alpha_value_set(ma_alpha)
+        ma_wrap.reflect_factor_set(ma_refl_factor)
+        ma_wrap.reflect_color_set(ma_refl_color)
+
+        cycles_material_wrap_map[ma] = ma_wrap
+    else:
+        # TODO, number BumpFactor isnt used yet
+        ma.diffuse_color = ma_diff
+        ma.specular_color = ma_spec
+        ma.alpha = ma_alpha
+        ma.specular_intensity = ma_spec_intensity
+        ma.specular_hardness = ma_spec_hardness * 5.10 + 1.0
+
+        if ma_refl_factor != 0.0:
+            ma.raytrace_mirror.use = True
+            ma.raytrace_mirror.reflect_factor = ma_refl_factor
+            ma.mirror_color = ma_refl_color
+
+    ma.use_fake_user = 1
+    return ma
+
+
+# -------
+# Texture
+
+def blen_read_texture(fbx_obj, basedir, image_cache,
+                      use_image_search):
+    import os
+    from bpy_extras import image_utils
+
+    elem_name, elem_class = elem_split_name_class(fbx_obj)
+    assert(elem_class == b'Texture')
+    elem_name_utf8 = elem_name.decode('utf-8')
+
+    filepath = elem_find_first_string(fbx_obj, b'FileName')
+    if os.sep == '/':
+        filepath = filepath.replace('\\', '/')
+    else:
+        filepath = filepath.replace('/', '\\')
+
+    image = image_cache.get(filepath)
+    if image is not None:
+        return image
+
+    image = image_utils.load_image(
+        filepath,
+        dirname=basedir,
+        place_holder=True,
+        recursive=use_image_search,
+        )
+
+    image.name = elem_name_utf8
+
+    return image
+
+
+def load(operator, context, filepath="",
+         global_matrix=None,
+         use_cycles=True,
+         use_image_search=False):
+
+    import os
+    from . import parse_fbx
+
+    try:
+        elem_root, version = parse_fbx.parse(filepath)
+    except:
+        import traceback
+        traceback.print_exc()
+
+        operator.report({'ERROR'}, "Couldn't open file %r" % filepath)
+        return {'CANCELLED'}
+
+    if version < 7100:
+        operator.report({'ERROR'}, "Version %r unsupported, must be %r or later" % (version, 7100))
+        return {'CANCELLED'}
+
+    # deselect all
+    if bpy.ops.object.select_all.poll():
+        bpy.ops.object.select_all(action='DESELECT')
+
+    basedir = os.path.dirname(filepath)
+
+    cycles_material_wrap_map = {}
+    image_cache = {}
+    if not use_cycles:
+        texture_cache = {}
+
+    # Tables: (FBX_byte_id -> [FBX_data, None or Blender_datablock])
+    fbx_table_nodes = {}
+
+    scene = context.scene
+
+    fbx_nodes = elem_find_first(elem_root, b'Objects')
+    fbx_connections = elem_find_first(elem_root, b'Connections')
+
+    if fbx_nodes is None:
+        return print("no 'Objects' found")
+    if fbx_connections is None:
+        return print("no 'Connections' found")
+
+    def _():
+        for fbx_obj in fbx_nodes.elems:
+            assert(fbx_obj.props_type == b'LSS')
+            fbx_uuid = elem_uuid(fbx_obj)
+            fbx_table_nodes[fbx_uuid] = [fbx_obj, None]
+    _(); del _
+
+    # ----
+    # First load in the data
+    # http://download.autodesk.com/us/fbx/20112/FBX_SDK_HELP/index.html?url=WS73099cc142f487551fea285e1221e4f9ff8-7fda.htm,topicNumber=d0e6388
+
+    fbx_connection_map = {}
+    fbx_connection_map_reverse = {}
+
+    def _():
+        for fbx_link in fbx_connections.elems:
+            # print(fbx_link)
+            c_type = fbx_link.props[0]
+            c_src, c_dst = fbx_link.props[1:3]
+            # if c_type == b'OO':
+
+            fbx_connection_map.setdefault(c_src, []).append((c_dst, fbx_link))
+            fbx_connection_map_reverse.setdefault(c_dst, []).append((c_src, fbx_link))
+    _(); del _
+
+    # ----
+    # Load mesh data
+    def _():
+        for fbx_uuid, fbx_item in fbx_table_nodes.items():
+            fbx_obj, blen_data = fbx_item
+            if fbx_obj.id != b'Geometry':
+                continue
+            if fbx_obj.props[-1] == b'Mesh':
+                assert(blen_data is None)
+                fbx_item[1] = blen_read_geom(fbx_obj)
+    _(); del _
+
+    # ----
+    # Load material data
+    def _():
+        for fbx_uuid, fbx_item in fbx_table_nodes.items():
+            fbx_obj, blen_data = fbx_item
+            if fbx_obj.id != b'Material':
+                continue
+            assert(blen_data is None)
+            fbx_item[1] = blen_read_material(fbx_obj,
+                                             cycles_material_wrap_map, use_cycles)
+    _(); del _
+
+    # ----
+    # Load image data
+    def _():
+        for fbx_uuid, fbx_item in fbx_table_nodes.items():
+            fbx_obj, blen_data = fbx_item
+            if fbx_obj.id != b'Texture':
+                continue
+            fbx_item[1] = blen_read_texture(fbx_obj, basedir, image_cache,
+                                            use_image_search)
+    _(); del _
+
+    # ----
+    # Connections
+    def connection_filter_ex(fbx_uuid, fbx_id, dct):
+        return [(c_found[0], c_found[1], c_type)
+                for (c_uuid, c_type) in dct.get(fbx_uuid, ())
+                for c_found in (fbx_table_nodes[c_uuid],)
+                if c_found[0].id == fbx_id]
+
+    def connection_filter_forward(fbx_uuid, fbx_id):
+        return connection_filter_ex(fbx_uuid, fbx_id, fbx_connection_map)
+
+    def connection_filter_reverse(fbx_uuid, fbx_id):
+        return connection_filter_ex(fbx_uuid, fbx_id, fbx_connection_map_reverse)
+
+    def _():
+        # link Material's to Geometry (via Model's)
+        for fbx_uuid, fbx_item in fbx_table_nodes.items():
+            fbx_obj, blen_data = fbx_item
+            if fbx_obj.id != b'Geometry':
+                continue
+
+            mesh = fbx_table_nodes[fbx_uuid][1]
+            for fbx_lnk, fbx_lnk_item, fbx_lnk_type in connection_filter_forward(fbx_uuid, b'Model'):
+
+                # create when linking since we need object data
+                obj = blen_read_object(fbx_lnk, mesh)
+                # fbx_lnk_item[1] = obj
+
+                # instance in scene
+                # obj.matrix_world = global_matrix * obj.matrix_world
+                obj_base = scene.objects.link(obj)
+                obj_base.select = True
+
+                # link materials
+                fbx_lnk_uuid = elem_uuid(fbx_lnk)
+                for fbx_lnk_material, material, fbx_lnk_material_type in connection_filter_reverse(fbx_lnk_uuid, b'Material'):
+                    mesh.materials.append(material)
+    _(); del _
+
+    def _():
+        # textures that use this material
+        def texture_bumpfac_get(fbx_obj):
+            fbx_props = elem_find_first(fbx_obj, b'Properties70')
+            return elem_props_get_number(fbx_props, b'BumpFactor', 1.0)
+
+        for fbx_uuid, fbx_item in fbx_table_nodes.items():
+            fbx_obj, blen_data = fbx_item
+            if fbx_obj.id != b'Material':
+                continue
+
+            material = fbx_table_nodes[fbx_uuid][1]
+            for fbx_lnk, image, fbx_lnk_type in connection_filter_reverse(fbx_uuid, b'Texture'):
+                if use_cycles:
+                    if fbx_lnk_type.props[0] == b'OP':
+                        lnk_type = fbx_lnk_type.props[3]
+
+                        ma_wrap = cycles_material_wrap_map[material]
+
+                        if lnk_type == b'DiffuseColor':
+                            ma_wrap.diffuse_image_set(image)
+                        elif lnk_type == b'SpecularColor':
+                            ma_wrap.specular_image_set(image)
+                        elif lnk_type == b'ReflectionColor':
+                            ma_wrap.reflect_image_set(image)
+                        elif lnk_type == b'TransparentColor':
+                            ma_wrap.alpha_image_set(image)
+                        elif lnk_type == b'DiffuseFactor':
+                            pass  # TODO
+                        elif lnk_type == b'ShininessExponent':
+                            ma_wrap.hardness_image_set(image)
+                        elif lnk_type == b'NormalMap':
+                            ma_wrap.normal_image_set(image)
+                            ma_wrap.normal_factor_set(texture_bumpfac_get(fbx_obj))
+                        elif lnk_type == b'Bump':
+                            ma_wrap.bump_image_set(image)
+                            ma_wrap.bump_factor_set(texture_bumpfac_get(fbx_obj))
+                else:
+                    if fbx_lnk_type.props[0] == b'OP':
+                        lnk_type = fbx_lnk_type.props[3]
+
+                        # cache converted texture
+                        tex = texture_cache.get(image)
+                        if tex is None:
+                            tex = bpy.data.textures.new(name=image.name, type='IMAGE')
+                            tex.image = image
+                            texture_cache[image] = tex
+
+                        mtex = material.texture_slots.add()
+                        mtex.texture = tex
+                        mtex.texture_coords = 'UV'
+                        mtex.use_map_color_diffuse = False
+
+                        if lnk_type == b'DiffuseColor':
+                            mtex.use_map_color_diffuse = True
+                            mtex.blend_type = 'MULTIPLY'
+                        elif lnk_type == b'SpecularColor':
+                            mtex.use_map_color_spec = True
+                            mtex.blend_type = 'MULTIPLY'
+                        elif lnk_type == b'ReflectionColor':
+                            mtex.use_map_raymir = True
+                        elif lnk_type == b'TransparentColor':
+                            pass
+                        elif lnk_type == b'DiffuseFactor':
+                            mtex.use_map_diffuse = True
+                        elif lnk_type == b'ShininessExponent':
+                            mtex.use_map_hardness = True
+                        elif lnk_type == b'NormalMap':
+                            tex.use_normal_map = True  # not ideal!
+                            mtex.use_map_normal = True
+                            mtex.normal_factor = texture_bumpfac_get(fbx_obj)
+                        elif lnk_type == b'Bump':
+                            mtex.use_map_normal = True
+                            mtex.normal_factor = texture_bumpfac_get(fbx_obj)
+                        else:
+                            print("WARNING: material link %r ignored" % lnk_type)
+    _(); del _
+
+    # print(list(sorted(locals().keys())))
+
+    return {'FINISHED'}
diff --git a/io_scene_fbx/parse_fbx.py b/io_scene_fbx/parse_fbx.py
new file mode 100644
index 0000000000000000000000000000000000000000..922bc4e49cc9f28177485abc6fa780a80187b3a8
--- /dev/null
+++ b/io_scene_fbx/parse_fbx.py
@@ -0,0 +1,176 @@
+# ##### BEGIN GPL LICENSE BLOCK #####
+#
+#  This program is free software; you can redistribute it and/or
+#  modify it under the terms of the GNU General Public License
+#  as published by the Free Software Foundation; either version 2
+#  of the License, or (at your option) any later version.
+#
+#  This program is distributed in the hope that it will be useful,
+#  but WITHOUT ANY WARRANTY; without even the implied warranty of
+#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+#  GNU General Public License for more details.
+#
+#  You should have received a copy of the GNU General Public License
+#  along with this program; if not, write to the Free Software Foundation,
+#  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+#
+# ##### END GPL LICENSE BLOCK #####
+
+# <pep8 compliant>
+
+# Script copyright (C) 2006-2012, assimp team
+# Script copyright (C) 2013 Blender Foundation
+
+__all__ = (
+    "parse",
+    "data_types",
+    "FBXElem",
+    )
+
+from struct import unpack
+import array
+import zlib
+
+# at the end of each nested block, there is a NUL record to indicate
+# that the sub-scope exists (i.e. to distinguish between P: and P : {})
+# this NUL record is 13 bytes long.
+_BLOCK_SENTINEL_LENGTH = 13
+_BLOCK_SENTINEL_DATA = (b'\0' * _BLOCK_SENTINEL_LENGTH)
+_IS_BIG_ENDIAN = (__import__("sys").byteorder != 'little')
+from collections import namedtuple
+FBXElem = namedtuple("FBXElem", ("id", "props", "props_type", "elems"))
+del namedtuple
+
+
+def read_uint(read):
+    return unpack(b'<I', read(4))[0]
+
+
+def read_ubyte(read):
+    return unpack(b'B', read(1))[0]
+
+
+def read_string_ubyte(read):
+    size = read_ubyte(read)
+    data = read(size)
+    return data
+
+
+def unpack_array(read, array_type, array_stride, array_byteswap):
+    length = read_uint(read)
+    encoding = read_uint(read)
+    comp_len = read_uint(read)
+
+    data = read(comp_len)
+
+    if encoding == 0:
+        pass
+    elif encoding == 1:
+        data = zlib.decompress(data)
+
+    assert(length * array_stride == len(data))
+
+    data_array = array.array(array_type, data)
+    if array_byteswap and _IS_BIG_ENDIAN:
+        data_array.byteswap()
+    return data_array
+
+
+read_data_dict = {
+    b'Y'[0]: lambda read, size: unpack(b'<h', read(2))[0],  # 16 bit int
+    b'C'[0]: lambda read, size: unpack(b'?', read(1))[0],   # 1 bit bool (yes/no)
+    b'I'[0]: lambda read, size: unpack(b'<i', read(4))[0],  # 32 bit int
+    b'F'[0]: lambda read, size: unpack(b'<f', read(4))[0],  # 32 bit float
+    b'D'[0]: lambda read, size: unpack(b'<d', read(8))[0],  # 64 bit float
+    b'L'[0]: lambda read, size: unpack(b'<q', read(8))[0],  # 64 bit int
+    b'R'[0]: lambda read, size: read(read_uint(read)),      # binary data
+    b'S'[0]: lambda read, size: read(read_uint(read)),      # string data
+    b'f'[0]: lambda read, size: unpack_array(read, 'f', 4, False),  # array (float)
+    b'i'[0]: lambda read, size: unpack_array(read, 'i', 4, True),   # array (int)
+    b'd'[0]: lambda read, size: unpack_array(read, 'd', 8, False),  # array (double)
+    b'l'[0]: lambda read, size: unpack_array(read, 'q', 8, True),   # array (long)
+    b'b'[0]: lambda read, size: read(size),  # unknown
+    }
+
+
+def read_elem(read, tell, use_namedtuple):
+    # [0] the offset at which this block ends
+    # [1] the number of properties in the scope
+    # [2] the length of the property list
+    end_offset = read_uint(read)
+    if end_offset == 0:
+        return None
+
+    prop_count = read_uint(read)
+    prop_length = read_uint(read)
+
+    elem_id = read_string_ubyte(read)        # elem name of the scope/key
+    elem_props_type = bytearray(prop_count)  # elem property types
+    elem_props_data = [None] * prop_count    # elem properties (if any)
+    elem_subtree = []                        # elem children (if any)
+
+    for i in range(prop_count):
+        data_type = read(1)[0]
+        elem_props_data[i] = read_data_dict[data_type](read, prop_length)
+        elem_props_type[i] = data_type
+
+    if tell() < end_offset:
+        while tell() < (end_offset - _BLOCK_SENTINEL_LENGTH):
+            elem_subtree.append(read_elem(read, tell, use_namedtuple))
+
+        if read(_BLOCK_SENTINEL_LENGTH) != _BLOCK_SENTINEL_DATA:
+            raise IOError("failed to read nested block sentinel, "
+                          "expected all bytes to be 0")
+
+    if tell() != end_offset:
+        raise IOError("scope length not reached, something is wrong")
+
+    args = (elem_id, elem_props_data, elem_props_type, elem_subtree)
+    return FBXElem(*args) if use_namedtuple else args
+
+
+def parse(fn, use_namedtuple=True):
+    # import time
+    # t = time.time()
+
+    root_elems = []
+
+    with open(fn, 'rb') as f:
+        read = f.read
+        tell = f.tell
+
+        HEAD_MAGIC = b'Kaydara FBX Binary\x20\x20\x00\x1a\x00'
+        if read(len(HEAD_MAGIC)) != HEAD_MAGIC:
+            raise IOError("Invalid header")
+
+        fbx_version = read_uint(read)
+
+        while True:
+            elem = read_elem(read, tell, use_namedtuple)
+            if elem is None:
+                break
+            root_elems.append(elem)
+
+    # print("done in %.4f sec" % (time.time() - t))
+
+    args = (b'', [], bytearray(0), root_elems)
+    return FBXElem(*args) if use_namedtuple else args, fbx_version
+
+# Inline module, only for external use
+# pyfbx.data_types
+data_types = type(array)("data_types")
+data_types.__dict__.update(
+dict(
+INT16 = b'Y'[0],
+BOOL = b'C'[0],
+INT32 = b'I'[0],
+FLOAT32 = b'F'[0],
+FLOAT64 = b'D'[0],
+INT64 = b'L'[0],
+BYTES = b'R'[0],
+STRING = b'S'[0],
+FLOAT32_ARRAY = b'f'[0],
+INT32_ARRAY = b'i'[0],
+FLOAT64_ARRAY = b'd'[0],
+INT64_ARRAY = b'l'[0],
+))