Frameworks and Weak Linking

来源:互联网 发布:ubuntu wine 启动 编辑:程序博客网 时间:2024/05/07 18:11

Frameworks and Weak Linking

One challenge faced by developers is that of taking advantage of new features introduced in new versions of Mac OS X while still supporting older versions of the system. Normally, if an application uses a new feature in a framework, it is unable to run on earlier versions of the framework that do not support that feature. Such applications would either fail to launch or crash when an attempt to use the feature was made. Apple has solved this problem by adding support for weakly-linked symbols.

When a symbol in a framework is defined as weakly linked, the symbol does not have to be present at runtime for a process to continue running. The static linker identifies a weakly linked symbol as such in any code module that references the symbol. The dynamic linker uses this same information at runtime to determine whether a process can continue running. If a weakly linked symbol is not present in the framework, the code module can continue to run as long as it does not reference the symbol. However, if the symbol is present, the code can use it normally.

If you are updating your own frameworks, you should consider making new symbols weakly linked. Doing so can make it easier for clients of your framework to support it. You should also make sure that your own code checks for the existence of weakly-linked symbols before using them.

Note: Although the Code Fragment Manager supports its own form of weak linking, the information that follows pertains solely to Mach-O executables.

For more information regarding weak linking, including additional examples, see SDK Compatibility Guide.

Weak Linking and Apple Frameworks

Apple frameworks use the availability macros to determine whether a symbol is weakly linked or strongly linked. Apple wraps new interfaces in its frameworks with availability macros to indicate which version of the operating system a feature first appeared. Macros are also used to indicate deprecated features and interfaces.

The availability macros defined in /usr/include/AvailabilityMacros.h add weak linking information to system interfaces based on the versions of Mac OS X your project supports. When you create a new project, you tell the compiler which versions of Mac OS X your project supports by setting the deployment target and target SDK in Xcode. The compiler uses these settings to assign appropriate values to the MAC_OS_X_VERSION_MIN_REQUIRED andMAC_OS_X_VERSION_MAX_ALLOWED macros, respectively. For information on how to modify these settings in Xcode, see “Setting Up Cross-Development in Xcode” in SDK Compatibility Guide or the Xcode help.

For example, suppose in Xcode you set the deployment target (minimum required version) to Mac OS X 10.2 and the target SDK (maximum allowed version) to Mac OS X 10.3. During compilation, the compiler would weakly link any interfaces that were introduced in Mac OS X version 10.3 while strongly linking earlier interfaces. This would allow your application to continue running on Mac OS X version 10.2 but still take advantage of newer features when they are available.

Important The deployment target setting in Xcode must be set to Mac OS X version 10.2 or later to take advantage of weak linking. If you do not set this value (or set it to an earlier version of Mac OS X), you cannot use weak linking in your project.

Before using any symbols that are introduced in a version of Mac OS X that is later than your minimum required version, make sure you check to see that the symbol exists first. See “Using Weakly Linked Symbols” for more information.

Marking Symbols for Weak Linking

If you define your own frameworks, you can mark symbols as weakly linked using the weak_import attribute. Weak linking is especially appropriate if you introduce new features to an existing framework. To mark symbols as weakly linked, you must make sure your environment is configured to support weak linking:

  • You must be using GCC version 3.1 or later. Weak linking is not supported in GCC version 2.95

  • You must set the Mac OS X Deployment Target build option of your Xcode project to Mac OS X 10.2 or later.

The linker marks symbols as strongly linked unless you explicitly tell it otherwise. To mark a function or variable as weakly linked, add the weak_import attribute to the function prototype or variable declaration, as shown in the following example:

extern int MyFunction() __attribute__((weak_import));
extern int MyVariable __attribute__((weak_import));

Using Weakly Linked Symbols

If your framework relies on weakly linked symbols in any Apple or third-party frameworks, you must check for the existence of those symbols before using them. If you attempt to use a non-existent symbol without first checking, the dynamic linker may generate a runtime binding error and terminate the corresponding process.

If a weakly linked symbol is not available in a framework, the linker sets the address of the symbol to NULL. You can check this address in your code using code similar to the following:

extern int MyWeakLinkedFunction() __attribute__((weak_import));
 
int main()
{
    int result = 0;
 
    if (MyWeakLinkedFunction != NULL)
    {
        result = MyWeakLinkedFunction();
    }
 
    return result;
}

Note: When checking for the existence of a symbol, you must explicitly compare it to NULL or nil in your code. You cannot use the negation operator ( ! ) to negate the address of the symbol.

Weak Linking to Entire Frameworks

When you reference symbols in another framework, most of those symbols are linked strongly to your code. In order to create a weak link to a symbol, the framework containing the symbol must explicitly add the weak_import attribute to it (see “Marking Symbols for Weak Linking”). However, if you do not maintain a framework and need to link its symbols weakly, you can explicitly tell the compiler to mark all symbols as weakly linked. To do this, you must open your project in Xcode and modify the way your targets link to the framework as follows:

  1. Select the target you want to modify and reveal its build phases.

  2. Expand the Link Binary With Libraries build phase to view the frameworks currently linked by the target.

  3. If the framework you want to weakly link to is listed in the Link Binary With Libraries build phase, select it, and choose Edit > Delete to remove it.

    Now you can tell the linker to use weak linking for that framework.

  4. Select the target, open its Info window, and click Build.

  5. To the Other Linker Flags build setting, add the following command-line option specification, where <framework_name> is the name of the framework you want to weakly link to:

    -weak_framework <framework_name>
  6. Build your product.

The -weak_framework option tells the linker to weakly link all symbols in the named framework. If you need to link to a library instead of a framework, you can use the -weak_library linker command instead of -weak_framework.

原创粉丝点击