199451b44SJordan Rupprecht
299451b44SJordan Rupprechtimport unittest2
399451b44SJordan Rupprechtimport os
499451b44SJordan Rupprechtimport shutil
599451b44SJordan Rupprecht
699451b44SJordan Rupprechtimport lldb
799451b44SJordan Rupprechtfrom lldbsuite.test.decorators import *
899451b44SJordan Rupprechtfrom lldbsuite.test.lldbtest import *
999451b44SJordan Rupprechtfrom lldbsuite.test import lldbutil
1099451b44SJordan Rupprecht
1199451b44SJordan Rupprecht
1299451b44SJordan Rupprechtclass TestClangModuleUpdate(TestBase):
1399451b44SJordan Rupprecht
1499451b44SJordan Rupprecht    @skipIf(debug_info=no_match(["gmodules"]))
15*accb0955SJonas Devlieghere    @skipIfDarwin # rdar://76540904
1699451b44SJordan Rupprecht    def test_expr(self):
1799451b44SJordan Rupprecht        with open(self.getBuildArtifact("module.modulemap"), "w") as f:
1899451b44SJordan Rupprecht            f.write("""
1999451b44SJordan Rupprecht                    module Foo { header "f.h" }
2099451b44SJordan Rupprecht                    """)
2199451b44SJordan Rupprecht        with open(self.getBuildArtifact("f.h"), "w") as f:
2299451b44SJordan Rupprecht            f.write("""
2399451b44SJordan Rupprecht                    struct Q { int i; };
2499451b44SJordan Rupprecht                    void f() {}
2599451b44SJordan Rupprecht                    """)
2699451b44SJordan Rupprecht
2799451b44SJordan Rupprecht        mod_cache = self.getBuildArtifact("private-module-cache")
2899451b44SJordan Rupprecht        if os.path.isdir(mod_cache):
2999451b44SJordan Rupprecht          shutil.rmtree(mod_cache)
3099451b44SJordan Rupprecht        d = {'OBJC_SOURCES': 'first.m'}
3199451b44SJordan Rupprecht        self.build(dictionary=d)
3299451b44SJordan Rupprecht        self.assertTrue(os.path.isdir(mod_cache), "module cache exists")
3399451b44SJordan Rupprecht
3499451b44SJordan Rupprecht        logfile = self.getBuildArtifact("modules.log")
3599451b44SJordan Rupprecht        self.runCmd("log enable -f %s lldb module" % logfile)
3699451b44SJordan Rupprecht        target, process, _, bkpt = lldbutil.run_to_name_breakpoint(self, "main")
3799451b44SJordan Rupprecht        self.assertIn("int i", str(target.FindTypes('Q').GetTypeAtIndex(0)))
3899451b44SJordan Rupprecht        self.expect("image list -g", patterns=[r'first\.o', r'Foo.*\.pcm'])
3999451b44SJordan Rupprecht
4099451b44SJordan Rupprecht        # Update the module.
4199451b44SJordan Rupprecht        with open(self.getBuildArtifact("f.h"), "w") as f:
4299451b44SJordan Rupprecht            f.write("""
4399451b44SJordan Rupprecht                    struct S { int i; };
4499451b44SJordan Rupprecht                    struct S getS() { struct S r = {1}; return r; }
4599451b44SJordan Rupprecht                    void f() {}
4699451b44SJordan Rupprecht                    """)
4799451b44SJordan Rupprecht
4899451b44SJordan Rupprecht        # Rebuild.
4999451b44SJordan Rupprecht        d = {'OBJC_SOURCES': 'second.m'}
5099451b44SJordan Rupprecht        self.build(dictionary=d)
5199451b44SJordan Rupprecht
5299451b44SJordan Rupprecht        # Reattach.
5399451b44SJordan Rupprecht        process.Kill()
5499451b44SJordan Rupprecht        target, process, _, _ = lldbutil.run_to_breakpoint_do_run(self, target, bkpt)
5599451b44SJordan Rupprecht        self.assertIn("int i", str(target.FindTypes('S').GetTypeAtIndex(0)))
5699451b44SJordan Rupprecht        self.expect("image list -g", patterns=[r'second\.o', r'Foo.*\.pcm'])
5799451b44SJordan Rupprecht
5899451b44SJordan Rupprecht        # Check log file.
5999451b44SJordan Rupprecht        found = False
6099451b44SJordan Rupprecht        with open(logfile, 'r') as f:
6199451b44SJordan Rupprecht            for line in f:
6299451b44SJordan Rupprecht                if "module changed" in line and "Foo" in line:
6399451b44SJordan Rupprecht                    found = True
6499451b44SJordan Rupprecht        self.assertTrue(found)
65